blob: 6fa32b320a94f31a61f8294dc1a857b6527f332f [file] [log] [blame]
Keith Whitwell6dc85572003-07-17 13:43:59 +00001/**
2 * \file context.c
3 * Mesa context/visual/framebuffer management functions.
4 * \author Brian Paul
5 */
6
jtgafb833d1999-08-19 00:55:39 +00007/*
8 * Mesa 3-D graphics library
Brian Paul363344f2005-09-13 14:48:28 +00009 * Version: 6.5
jtgafb833d1999-08-19 00:55:39 +000010 *
Brian Paul2aabdc72006-02-24 18:19:11 +000011 * Copyright (C) 1999-2006 Brian Paul All Rights Reserved.
jtgafb833d1999-08-19 00:55:39 +000012 *
13 * Permission is hereby granted, free of charge, to any person obtaining a
14 * copy of this software and associated documentation files (the "Software"),
15 * to deal in the Software without restriction, including without limitation
16 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
17 * and/or sell copies of the Software, and to permit persons to whom the
18 * Software is furnished to do so, subject to the following conditions:
19 *
20 * The above copyright notice and this permission notice shall be included
21 * in all copies or substantial portions of the Software.
22 *
23 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
24 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
25 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
26 * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
27 * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
28 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 */
30
31
Keith Whitwell6dc85572003-07-17 13:43:59 +000032/**
Brian Paul253204f2004-09-10 00:45:12 +000033 * \mainpage Mesa Main Module
Keith Whitwell6dc85572003-07-17 13:43:59 +000034 *
Brian Paul253204f2004-09-10 00:45:12 +000035 * \section MainIntroduction Introduction
Keith Whitwell6dc85572003-07-17 13:43:59 +000036 *
Brian Paul253204f2004-09-10 00:45:12 +000037 * The Mesa Main module consists of all the files in the main/ directory.
38 * Among the features of this module are:
39 * <UL>
40 * <LI> Structures to represent most GL state </LI>
41 * <LI> State set/get functions </LI>
42 * <LI> Display lists </LI>
43 * <LI> Texture unit, object and image handling </LI>
44 * <LI> Matrix and attribute stacks </LI>
45 * </UL>
Keith Whitwell6dc85572003-07-17 13:43:59 +000046 *
Brian Paul253204f2004-09-10 00:45:12 +000047 * Other modules are responsible for API dispatch, vertex transformation,
48 * point/line/triangle setup, rasterization, vertex array caching,
49 * vertex/fragment programs/shaders, etc.
Keith Whitwell6dc85572003-07-17 13:43:59 +000050 *
51 *
52 * \section AboutDoxygen About Doxygen
53 *
54 * If you're viewing this information as Doxygen-generated HTML you'll
55 * see the documentation index at the top of this page.
56 *
57 * The first line lists the Mesa source code modules.
58 * The second line lists the indexes available for viewing the documentation
59 * for each module.
60 *
61 * Selecting the <b>Main page</b> link will display a summary of the module
62 * (this page).
63 *
Jose Fonseca1a5709d2003-09-17 17:14:11 +000064 * Selecting <b>Data Structures</b> will list all C structures.
Keith Whitwell6dc85572003-07-17 13:43:59 +000065 *
66 * Selecting the <b>File List</b> link will list all the source files in
67 * the module.
68 * Selecting a filename will show a list of all functions defined in that file.
69 *
Jose Fonseca1a5709d2003-09-17 17:14:11 +000070 * Selecting the <b>Data Fields</b> link will display a list of all
Keith Whitwell6dc85572003-07-17 13:43:59 +000071 * documented structure members.
72 *
Jose Fonseca1a5709d2003-09-17 17:14:11 +000073 * Selecting the <b>Globals</b> link will display a list
Keith Whitwell6dc85572003-07-17 13:43:59 +000074 * of all functions, structures, global variables and macros in the module.
75 *
76 */
77
78
Brian Paulfbd8f211999-11-11 01:22:25 +000079#include "glheader.h"
Brian Paul3c634522002-10-24 23:57:19 +000080#include "imports.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +000081#include "accum.h"
Brian Paulc04bb512006-07-11 21:56:43 +000082#include "arrayobj.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +000083#include "attrib.h"
84#include "blend.h"
Brian Paulb1394fa2000-09-26 20:53:53 +000085#include "buffers.h"
Brian Paul148a2842003-09-17 03:40:11 +000086#include "bufferobj.h"
Brian Paul4bdcfe52000-04-17 17:57:04 +000087#include "colortab.h"
jtgafb833d1999-08-19 00:55:39 +000088#include "context.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +000089#include "debug.h"
90#include "depth.h"
jtgafb833d1999-08-19 00:55:39 +000091#include "dlist.h"
92#include "eval.h"
93#include "enums.h"
Brian Paul585a68c1999-09-11 11:31:34 +000094#include "extensions.h"
Brian Paule4b23562005-05-04 20:11:35 +000095#include "fbobject.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +000096#include "feedback.h"
jtgafb833d1999-08-19 00:55:39 +000097#include "fog.h"
Briane6a93812007-02-26 11:37:37 -070098#include "framebuffer.h"
Brian Paulb7a43041999-11-30 20:34:51 +000099#include "get.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000100#include "histogram.h"
101#include "hint.h"
jtgafb833d1999-08-19 00:55:39 +0000102#include "hash.h"
103#include "light.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000104#include "lines.h"
jtgafb833d1999-08-19 00:55:39 +0000105#include "macros.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000106#include "matrix.h"
107#include "pixel.h"
108#include "points.h"
109#include "polygon.h"
Brian74afcab2007-04-21 12:42:54 -0600110#include "queryobj.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000111#include "rastpos.h"
jtgafb833d1999-08-19 00:55:39 +0000112#include "simple_list.h"
Brian Paulfa9df402000-02-02 19:16:46 +0000113#include "state.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000114#include "stencil.h"
Brian Paul8f04c122004-04-27 13:39:20 +0000115#include "texcompress.h"
jtgafb833d1999-08-19 00:55:39 +0000116#include "teximage.h"
117#include "texobj.h"
Brian Paul85d81602002-06-17 23:36:31 +0000118#include "texstate.h"
Jouk Jansen5e3bc0c2000-11-22 07:32:16 +0000119#include "mtypes.h"
jtgafb833d1999-08-19 00:55:39 +0000120#include "varray.h"
Brian Paul363344f2005-09-13 14:48:28 +0000121#include "version.h"
Gareth Hughesd4eb6652001-03-12 01:32:20 +0000122#include "vtxfmt.h"
Brianc223c6b2007-07-04 13:15:20 -0600123#include "glapi/glthread.h"
124#include "glapi/glapioffsets.h"
125#if FEATURE_NV_vertex_program || FEATURE_NV_fragment_program
126#include "shader/program.h"
127#endif
128#include "shader/shader_api.h"
129#include "shader/atifragshader.h"
Keith Whitwell6dc85572003-07-17 13:43:59 +0000130#if _HAVE_FULL_GL
Keith Whitwell23caf202000-11-16 21:05:34 +0000131#include "math/m_translate.h"
Keith Whitwell23caf202000-11-16 21:05:34 +0000132#include "math/m_matrix.h"
133#include "math/m_xform.h"
Keith Whitwellf4b02d12001-01-05 05:31:42 +0000134#include "math/mathmod.h"
Brian Paulddc82ee2005-02-05 19:56:45 +0000135#endif
jtgafb833d1999-08-19 00:55:39 +0000136
davem69775355a2001-06-05 23:54:00 +0000137#ifdef USE_SPARC_ASM
Ian Romanicke16f6e32004-06-26 00:02:51 +0000138#include "sparc/sparc.h"
davem69775355a2001-06-05 23:54:00 +0000139#endif
jtgafb833d1999-08-19 00:55:39 +0000140
Keith Whitwell23caf202000-11-16 21:05:34 +0000141#ifndef MESA_VERBOSE
Keith Whitwell306d3fc2002-04-09 16:56:50 +0000142int MESA_VERBOSE = 0;
Keith Whitwell23caf202000-11-16 21:05:34 +0000143#endif
144
145#ifndef MESA_DEBUG_FLAGS
Keith Whitwell306d3fc2002-04-09 16:56:50 +0000146int MESA_DEBUG_FLAGS = 0;
Keith Whitwell23caf202000-11-16 21:05:34 +0000147#endif
Brian Paulb1394fa2000-09-26 20:53:53 +0000148
Brian Paul86b84272001-12-14 02:50:01 +0000149
Brian Paul27558a12003-03-01 01:50:20 +0000150/* ubyte -> float conversion */
151GLfloat _mesa_ubyte_to_float_color_tab[256];
152
Brian Paul9a33a112002-06-13 04:28:29 +0000153static void
154free_shared_state( GLcontext *ctx, struct gl_shared_state *ss );
155
Brian Paul86b84272001-12-14 02:50:01 +0000156
Keith Whitwell6dc85572003-07-17 13:43:59 +0000157/**
158 * Swap buffers notification callback.
159 *
160 * \param gc GL context.
161 *
162 * Called by window system just before swapping buffers.
Brian Paul9a33a112002-06-13 04:28:29 +0000163 * We have to finish any pending rendering.
164 */
165void
166_mesa_notifySwapBuffers(__GLcontext *gc)
167{
168 FLUSH_VERTICES( gc, 0 );
169}
170
Brian Paulb1394fa2000-09-26 20:53:53 +0000171
jtgafb833d1999-08-19 00:55:39 +0000172/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +0000173/** \name GL Visual allocation/destruction */
Brian Paul4d053dd2000-01-14 04:45:47 +0000174/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +0000175/*@{*/
Brian Paul4d053dd2000-01-14 04:45:47 +0000176
Keith Whitwell6dc85572003-07-17 13:43:59 +0000177/**
Brian Paul894844a2004-03-21 17:05:03 +0000178 * Allocates a GLvisual structure and initializes it via
179 * _mesa_initialize_visual().
Keith Whitwell6dc85572003-07-17 13:43:59 +0000180 *
181 * \param rgbFlag GL_TRUE for RGB(A) mode, GL_FALSE for Color Index mode.
182 * \param dbFlag double buffering
183 * \param stereoFlag stereo buffer
184 * \param depthBits requested bits per depth buffer value. Any value in [0, 32]
185 * is acceptable but the actual depth type will be GLushort or GLuint as
186 * needed.
187 * \param stencilBits requested minimum bits per stencil buffer value
188 * \param accumRedBits, accumGreenBits, accumBlueBits, accumAlphaBits number of bits per color component in accum buffer.
189 * \param indexBits number of bits per pixel if \p rgbFlag is GL_FALSE
190 * \param redBits number of bits per color component in frame buffer for RGB(A)
191 * mode. We always use 8 in core Mesa though.
192 * \param greenBits same as above.
193 * \param blueBits same as above.
194 * \param alphaBits same as above.
195 * \param numSamples not really used.
196 *
197 * \return pointer to new GLvisual or NULL if requested parameters can't be
198 * met.
199 *
Brian Paul894844a2004-03-21 17:05:03 +0000200 * \note Need to add params for level and numAuxBuffers (at least)
Brian Paul4d053dd2000-01-14 04:45:47 +0000201 */
Brian Paulb371e0d2000-03-31 01:05:51 +0000202GLvisual *
203_mesa_create_visual( GLboolean rgbFlag,
Brian Paulb371e0d2000-03-31 01:05:51 +0000204 GLboolean dbFlag,
205 GLboolean stereoFlag,
206 GLint redBits,
207 GLint greenBits,
208 GLint blueBits,
209 GLint alphaBits,
210 GLint indexBits,
211 GLint depthBits,
212 GLint stencilBits,
213 GLint accumRedBits,
214 GLint accumGreenBits,
215 GLint accumBlueBits,
216 GLint accumAlphaBits,
217 GLint numSamples )
Brian Paul4d053dd2000-01-14 04:45:47 +0000218{
Brian Paulc7e164f2006-06-30 15:44:30 +0000219 GLvisual *vis = (GLvisual *) _mesa_calloc(sizeof(GLvisual));
Brian Paul178a1c52000-04-22 01:05:00 +0000220 if (vis) {
Brian Paule70c6232000-05-04 13:53:55 +0000221 if (!_mesa_initialize_visual(vis, rgbFlag, dbFlag, stereoFlag,
Brian Paul178a1c52000-04-22 01:05:00 +0000222 redBits, greenBits, blueBits, alphaBits,
223 indexBits, depthBits, stencilBits,
224 accumRedBits, accumGreenBits,
225 accumBlueBits, accumAlphaBits,
Brian Paulb1394fa2000-09-26 20:53:53 +0000226 numSamples)) {
Brian Paulc7e164f2006-06-30 15:44:30 +0000227 _mesa_free(vis);
Brian Paul178a1c52000-04-22 01:05:00 +0000228 return NULL;
229 }
230 }
231 return vis;
232}
233
Keith Whitwell6dc85572003-07-17 13:43:59 +0000234/**
Brian Paul894844a2004-03-21 17:05:03 +0000235 * Makes some sanity checks and fills in the fields of the
Brian Paule4b23562005-05-04 20:11:35 +0000236 * GLvisual object with the given parameters. If the caller needs
237 * to set additional fields, he should just probably init the whole GLvisual
238 * object himself.
Keith Whitwell6dc85572003-07-17 13:43:59 +0000239 * \return GL_TRUE on success, or GL_FALSE on failure.
240 *
241 * \sa _mesa_create_visual() above for the parameter description.
Brian Paul178a1c52000-04-22 01:05:00 +0000242 */
243GLboolean
244_mesa_initialize_visual( GLvisual *vis,
245 GLboolean rgbFlag,
Brian Paul178a1c52000-04-22 01:05:00 +0000246 GLboolean dbFlag,
247 GLboolean stereoFlag,
248 GLint redBits,
249 GLint greenBits,
250 GLint blueBits,
251 GLint alphaBits,
252 GLint indexBits,
253 GLint depthBits,
254 GLint stencilBits,
255 GLint accumRedBits,
256 GLint accumGreenBits,
257 GLint accumBlueBits,
258 GLint accumAlphaBits,
259 GLint numSamples )
260{
261 assert(vis);
Brian Paul4d053dd2000-01-14 04:45:47 +0000262
Brian Pauled30dfa2000-03-03 17:47:39 +0000263 if (depthBits < 0 || depthBits > 32) {
Brian Paul178a1c52000-04-22 01:05:00 +0000264 return GL_FALSE;
Brian Paul4d053dd2000-01-14 04:45:47 +0000265 }
Brian Paule4b23562005-05-04 20:11:35 +0000266 if (stencilBits < 0 || stencilBits > STENCIL_BITS) {
Brian Paul178a1c52000-04-22 01:05:00 +0000267 return GL_FALSE;
Brian Paul4d053dd2000-01-14 04:45:47 +0000268 }
Brian Paul978ef2b2005-09-21 03:35:08 +0000269 assert(accumRedBits >= 0);
270 assert(accumGreenBits >= 0);
271 assert(accumBlueBits >= 0);
272 assert(accumAlphaBits >= 0);
Brian Paul4d053dd2000-01-14 04:45:47 +0000273
Brian Paulb6bcae52001-01-23 23:39:36 +0000274 vis->rgbMode = rgbFlag;
275 vis->doubleBufferMode = dbFlag;
276 vis->stereoMode = stereoFlag;
Brian Paul153f1542002-10-29 15:04:35 +0000277
Brian Paulb6bcae52001-01-23 23:39:36 +0000278 vis->redBits = redBits;
279 vis->greenBits = greenBits;
280 vis->blueBits = blueBits;
281 vis->alphaBits = alphaBits;
Brian Paule4b23562005-05-04 20:11:35 +0000282 vis->rgbBits = redBits + greenBits + blueBits;
Brian Paul4d053dd2000-01-14 04:45:47 +0000283
Brian Paulb6bcae52001-01-23 23:39:36 +0000284 vis->indexBits = indexBits;
285 vis->depthBits = depthBits;
Brian Paule4b23562005-05-04 20:11:35 +0000286 vis->stencilBits = stencilBits;
287
288 vis->accumRedBits = accumRedBits;
289 vis->accumGreenBits = accumGreenBits;
290 vis->accumBlueBits = accumBlueBits;
291 vis->accumAlphaBits = accumAlphaBits;
Brian Pauled30dfa2000-03-03 17:47:39 +0000292
Brian Paul153f1542002-10-29 15:04:35 +0000293 vis->haveAccumBuffer = accumRedBits > 0;
294 vis->haveDepthBuffer = depthBits > 0;
295 vis->haveStencilBuffer = stencilBits > 0;
296
297 vis->numAuxBuffers = 0;
298 vis->level = 0;
299 vis->pixmapMode = 0;
Brian Paule4b23562005-05-04 20:11:35 +0000300 vis->sampleBuffers = numSamples > 0 ? 1 : 0;
Brian Paul894844a2004-03-21 17:05:03 +0000301 vis->samples = numSamples;
Brian Paul153f1542002-10-29 15:04:35 +0000302
Brian Paul178a1c52000-04-22 01:05:00 +0000303 return GL_TRUE;
Brian Paul4d053dd2000-01-14 04:45:47 +0000304}
305
Brian Paul894844a2004-03-21 17:05:03 +0000306
Keith Whitwell6dc85572003-07-17 13:43:59 +0000307/**
Brian Paul894844a2004-03-21 17:05:03 +0000308 * Destroy a visual and free its memory.
Keith Whitwell6dc85572003-07-17 13:43:59 +0000309 *
310 * \param vis visual.
311 *
312 * Frees the visual structure.
313 */
Brian Paulb371e0d2000-03-31 01:05:51 +0000314void
315_mesa_destroy_visual( GLvisual *vis )
316{
Brian Paulc7e164f2006-06-30 15:44:30 +0000317 _mesa_free(vis);
Brian Paulb371e0d2000-03-31 01:05:51 +0000318}
319
Keith Whitwell6dc85572003-07-17 13:43:59 +0000320/*@}*/
321
Brian Paulb371e0d2000-03-31 01:05:51 +0000322
Brian Paul4d053dd2000-01-14 04:45:47 +0000323/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +0000324/** \name Context allocation, initialization, destroying
325 *
326 * The purpose of the most initialization functions here is to provide the
327 * default state values according to the OpenGL specification.
328 */
jtgafb833d1999-08-19 00:55:39 +0000329/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +0000330/*@{*/
jtgafb833d1999-08-19 00:55:39 +0000331
Keith Whitwell6dc85572003-07-17 13:43:59 +0000332/**
333 * One-time initialization mutex lock.
334 *
335 * \sa Used by one_time_init().
336 */
Brian Paul9560f052000-01-31 23:11:39 +0000337_glthread_DECLARE_STATIC_MUTEX(OneTimeLock);
338
Keith Whitwell6dc85572003-07-17 13:43:59 +0000339/**
340 * Calls all the various one-time-init functions in Mesa.
341 *
342 * While holding a global mutex lock, calls several initialization functions,
343 * and sets the glapi callbacks if the \c MESA_DEBUG environment variable is
344 * defined.
345 *
Brian Paula764b7e2006-02-26 17:16:37 +0000346 * \sa _math_init().
jtgafb833d1999-08-19 00:55:39 +0000347 */
Brian Paul178a1c52000-04-22 01:05:00 +0000348static void
Brian Paul4753d602002-06-15 02:38:15 +0000349one_time_init( GLcontext *ctx )
jtgafb833d1999-08-19 00:55:39 +0000350{
351 static GLboolean alreadyCalled = GL_FALSE;
Brian Paula6c423d2004-08-25 15:59:48 +0000352 (void) ctx;
Brian Paul9560f052000-01-31 23:11:39 +0000353 _glthread_LOCK_MUTEX(OneTimeLock);
jtgafb833d1999-08-19 00:55:39 +0000354 if (!alreadyCalled) {
Brian Paul27558a12003-03-01 01:50:20 +0000355 GLuint i;
356
Brian Paul4d053dd2000-01-14 04:45:47 +0000357 /* do some implementation tests */
358 assert( sizeof(GLbyte) == 1 );
Brian Paul4d053dd2000-01-14 04:45:47 +0000359 assert( sizeof(GLubyte) == 1 );
Brian Paul894844a2004-03-21 17:05:03 +0000360 assert( sizeof(GLshort) == 2 );
361 assert( sizeof(GLushort) == 2 );
362 assert( sizeof(GLint) == 4 );
363 assert( sizeof(GLuint) == 4 );
Brian Paul4d053dd2000-01-14 04:45:47 +0000364
Brian33c37392007-04-04 22:18:53 -0600365 _mesa_init_sqrt_table();
366
Keith Whitwell6dc85572003-07-17 13:43:59 +0000367#if _HAVE_FULL_GL
Keith Whitwell23caf202000-11-16 21:05:34 +0000368 _math_init();
Brian Paul27558a12003-03-01 01:50:20 +0000369
370 for (i = 0; i < 256; i++) {
371 _mesa_ubyte_to_float_color_tab[i] = (float) i / 255.0F;
372 }
Keith Whitwell6dc85572003-07-17 13:43:59 +0000373#endif
Brian Paul68ee4bc2000-01-28 19:02:22 +0000374
davem69775355a2001-06-05 23:54:00 +0000375#ifdef USE_SPARC_ASM
376 _mesa_init_sparc_glapi_relocs();
377#endif
Brian Paul3c634522002-10-24 23:57:19 +0000378 if (_mesa_getenv("MESA_DEBUG")) {
Brian Paul68ee4bc2000-01-28 19:02:22 +0000379 _glapi_noop_enable_warnings(GL_TRUE);
Brian Paul4e9676f2002-06-29 19:48:15 +0000380 _glapi_set_warning_func( (_glapi_warning_func) _mesa_warning );
Brian Paul68ee4bc2000-01-28 19:02:22 +0000381 }
382 else {
383 _glapi_noop_enable_warnings(GL_FALSE);
384 }
385
jtgafb833d1999-08-19 00:55:39 +0000386#if defined(DEBUG) && defined(__DATE__) && defined(__TIME__)
Brian Paul363344f2005-09-13 14:48:28 +0000387 _mesa_debug(ctx, "Mesa %s DEBUG build %s %s\n",
388 MESA_VERSION_STRING, __DATE__, __TIME__);
jtgafb833d1999-08-19 00:55:39 +0000389#endif
Brian Paul68ee4bc2000-01-28 19:02:22 +0000390
391 alreadyCalled = GL_TRUE;
392 }
Brian Paul9560f052000-01-31 23:11:39 +0000393 _glthread_UNLOCK_MUTEX(OneTimeLock);
jtgafb833d1999-08-19 00:55:39 +0000394}
395
Brian Paul894844a2004-03-21 17:05:03 +0000396
Keith Whitwell6dc85572003-07-17 13:43:59 +0000397/**
jtgafb833d1999-08-19 00:55:39 +0000398 * Allocate and initialize a shared context state structure.
Keith Whitwell6dc85572003-07-17 13:43:59 +0000399 * Initializes the display list, texture objects and vertex programs hash
400 * tables, allocates the texture objects. If it runs out of memory, frees
401 * everything already allocated before returning NULL.
Brian Paul894844a2004-03-21 17:05:03 +0000402 *
403 * \return pointer to a gl_shared_state structure on success, or NULL on
404 * failure.
jtgafb833d1999-08-19 00:55:39 +0000405 */
Brian Paula3f13702003-04-01 16:41:50 +0000406static GLboolean
407alloc_shared_state( GLcontext *ctx )
jtgafb833d1999-08-19 00:55:39 +0000408{
Brian Paula3f13702003-04-01 16:41:50 +0000409 struct gl_shared_state *ss = CALLOC_STRUCT(gl_shared_state);
jtgafb833d1999-08-19 00:55:39 +0000410 if (!ss)
Brian Paula3f13702003-04-01 16:41:50 +0000411 return GL_FALSE;
412
413 ctx->Shared = ss;
jtgafb833d1999-08-19 00:55:39 +0000414
Brian Paule4b684c2000-09-12 21:07:40 +0000415 _glthread_INIT_MUTEX(ss->Mutex);
jtgafb833d1999-08-19 00:55:39 +0000416
Brian Paule4b684c2000-09-12 21:07:40 +0000417 ss->DisplayList = _mesa_NewHashTable();
Brian Paulbb797902000-01-24 16:19:54 +0000418 ss->TexObjects = _mesa_NewHashTable();
Brian Paul451f3102003-04-17 01:48:19 +0000419#if FEATURE_NV_vertex_program || FEATURE_NV_fragment_program
Brian Paul610d5992003-01-14 04:55:45 +0000420 ss->Programs = _mesa_NewHashTable();
Brian Paul8dfc5b92002-10-16 17:57:51 +0000421#endif
jtgafb833d1999-08-19 00:55:39 +0000422
Brian Paul451f3102003-04-17 01:48:19 +0000423#if FEATURE_ARB_vertex_program
Brian Paul4d859f72004-01-23 18:57:05 +0000424 ss->DefaultVertexProgram = ctx->Driver.NewProgram(ctx, GL_VERTEX_PROGRAM_ARB, 0);
Brian Paul451f3102003-04-17 01:48:19 +0000425 if (!ss->DefaultVertexProgram)
426 goto cleanup;
427#endif
428#if FEATURE_ARB_fragment_program
Brian Paul4d859f72004-01-23 18:57:05 +0000429 ss->DefaultFragmentProgram = ctx->Driver.NewProgram(ctx, GL_FRAGMENT_PROGRAM_ARB, 0);
Brian Paul451f3102003-04-17 01:48:19 +0000430 if (!ss->DefaultFragmentProgram)
431 goto cleanup;
432#endif
Dave Airlie7f752fe2004-12-19 03:06:59 +0000433#if FEATURE_ATI_fragment_shader
Brian Paul63d68302005-11-19 16:43:04 +0000434 ss->ATIShaders = _mesa_NewHashTable();
435 ss->DefaultFragmentShader = _mesa_new_ati_fragment_shader(ctx, 0);
Dave Airlie7f752fe2004-12-19 03:06:59 +0000436 if (!ss->DefaultFragmentShader)
437 goto cleanup;
438#endif
Brian Paul451f3102003-04-17 01:48:19 +0000439
Tilman Sauerbeck17b50632006-07-11 19:03:21 +0000440#if FEATURE_ARB_vertex_buffer_object || FEATURE_ARB_pixel_buffer_object
Brian Paulddc82ee2005-02-05 19:56:45 +0000441 ss->BufferObjects = _mesa_NewHashTable();
Tilman Sauerbeck17b50632006-07-11 19:03:21 +0000442#endif
443
Ian Romanickee34e6e2006-06-12 16:26:29 +0000444 ss->ArrayObjects = _mesa_NewHashTable();
Brian Paulddc82ee2005-02-05 19:56:45 +0000445
Michal Krol365582d2006-08-01 20:07:31 +0000446#if FEATURE_ARB_shader_objects
Briana90046f2006-12-15 10:07:26 -0700447 ss->ShaderObjects = _mesa_NewHashTable();
Michal Krol365582d2006-08-01 20:07:31 +0000448#endif
Ian Romanick0207b472003-09-09 00:10:12 +0000449
Brian Paula3f13702003-04-01 16:41:50 +0000450 ss->Default1D = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_1D);
451 if (!ss->Default1D)
452 goto cleanup;
Brian Paula8523782000-11-19 23:10:25 +0000453
Brian Paula3f13702003-04-01 16:41:50 +0000454 ss->Default2D = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_2D);
455 if (!ss->Default2D)
456 goto cleanup;
jtgafb833d1999-08-19 00:55:39 +0000457
Brian Paula3f13702003-04-01 16:41:50 +0000458 ss->Default3D = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_3D);
459 if (!ss->Default3D)
460 goto cleanup;
Brian Paula8523782000-11-19 23:10:25 +0000461
Brian Paula3f13702003-04-01 16:41:50 +0000462 ss->DefaultCubeMap = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_CUBE_MAP_ARB);
463 if (!ss->DefaultCubeMap)
464 goto cleanup;
Brian Paula8523782000-11-19 23:10:25 +0000465
Brian Paula3f13702003-04-01 16:41:50 +0000466 ss->DefaultRect = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_RECTANGLE_NV);
467 if (!ss->DefaultRect)
468 goto cleanup;
Brian Paul413d6a22000-05-26 14:44:59 +0000469
Ian Romanickbb372f12007-05-16 15:34:22 -0700470 ss->Default1DArray = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_1D_ARRAY_EXT);
471 if (!ss->Default1DArray)
472 goto cleanup;
473
474 ss->Default2DArray = (*ctx->Driver.NewTextureObject)(ctx, 0, GL_TEXTURE_2D_ARRAY_EXT);
475 if (!ss->Default2DArray)
476 goto cleanup;
477
Brian Paula3f13702003-04-01 16:41:50 +0000478 /* Effectively bind the default textures to all texture units */
479 ss->Default1D->RefCount += MAX_TEXTURE_IMAGE_UNITS;
480 ss->Default2D->RefCount += MAX_TEXTURE_IMAGE_UNITS;
481 ss->Default3D->RefCount += MAX_TEXTURE_IMAGE_UNITS;
482 ss->DefaultCubeMap->RefCount += MAX_TEXTURE_IMAGE_UNITS;
483 ss->DefaultRect->RefCount += MAX_TEXTURE_IMAGE_UNITS;
Ian Romanickbb372f12007-05-16 15:34:22 -0700484 ss->Default1DArray->RefCount += MAX_TEXTURE_IMAGE_UNITS;
485 ss->Default2DArray->RefCount += MAX_TEXTURE_IMAGE_UNITS;
Brian Paula3f13702003-04-01 16:41:50 +0000486
Keith Whitwell5ac93f82006-11-01 14:21:57 +0000487 _glthread_INIT_MUTEX(ss->TexMutex);
488 ss->TextureStateStamp = 0;
489
490
Brian Paulddc82ee2005-02-05 19:56:45 +0000491#if FEATURE_EXT_framebuffer_object
492 ss->FrameBuffers = _mesa_NewHashTable();
493 if (!ss->FrameBuffers)
494 goto cleanup;
495 ss->RenderBuffers = _mesa_NewHashTable();
496 if (!ss->RenderBuffers)
497 goto cleanup;
498#endif
499
500
Brian Paula3f13702003-04-01 16:41:50 +0000501 return GL_TRUE;
502
503 cleanup:
504 /* Ran out of memory at some point. Free everything and return NULL */
505 if (ss->DisplayList)
506 _mesa_DeleteHashTable(ss->DisplayList);
507 if (ss->TexObjects)
508 _mesa_DeleteHashTable(ss->TexObjects);
509#if FEATURE_NV_vertex_program
510 if (ss->Programs)
511 _mesa_DeleteHashTable(ss->Programs);
512#endif
Brian Paul451f3102003-04-17 01:48:19 +0000513#if FEATURE_ARB_vertex_program
514 if (ss->DefaultVertexProgram)
Brian Paul4d859f72004-01-23 18:57:05 +0000515 ctx->Driver.DeleteProgram(ctx, ss->DefaultVertexProgram);
Brian Paul451f3102003-04-17 01:48:19 +0000516#endif
517#if FEATURE_ARB_fragment_program
518 if (ss->DefaultFragmentProgram)
Brian Paul4d859f72004-01-23 18:57:05 +0000519 ctx->Driver.DeleteProgram(ctx, ss->DefaultFragmentProgram);
Brian Paul451f3102003-04-17 01:48:19 +0000520#endif
Dave Airlie7f752fe2004-12-19 03:06:59 +0000521#if FEATURE_ATI_fragment_shader
522 if (ss->DefaultFragmentShader)
Brian Paul63d68302005-11-19 16:43:04 +0000523 _mesa_delete_ati_fragment_shader(ctx, ss->DefaultFragmentShader);
Dave Airlie7f752fe2004-12-19 03:06:59 +0000524#endif
Tilman Sauerbeck17b50632006-07-11 19:03:21 +0000525#if FEATURE_ARB_vertex_buffer_object || FEATURE_ARB_pixel_buffer_object
Ian Romanick0207b472003-09-09 00:10:12 +0000526 if (ss->BufferObjects)
527 _mesa_DeleteHashTable(ss->BufferObjects);
Brian Paulddc82ee2005-02-05 19:56:45 +0000528#endif
529
Tilman Sauerbeckc0eb7772006-07-11 19:36:27 +0000530 if (ss->ArrayObjects)
531 _mesa_DeleteHashTable (ss->ArrayObjects);
532
Michal Krol365582d2006-08-01 20:07:31 +0000533#if FEATURE_ARB_shader_objects
Brian0bf5dbe2006-12-19 18:02:41 -0700534 if (ss->ShaderObjects)
Briana90046f2006-12-15 10:07:26 -0700535 _mesa_DeleteHashTable (ss->ShaderObjects);
Michal Krol365582d2006-08-01 20:07:31 +0000536#endif
Ian Romanick0207b472003-09-09 00:10:12 +0000537
Brian Paulddc82ee2005-02-05 19:56:45 +0000538#if FEATURE_EXT_framebuffer_object
539 if (ss->FrameBuffers)
540 _mesa_DeleteHashTable(ss->FrameBuffers);
541 if (ss->RenderBuffers)
542 _mesa_DeleteHashTable(ss->RenderBuffers);
543#endif
544
Brian Paula3f13702003-04-01 16:41:50 +0000545 if (ss->Default1D)
546 (*ctx->Driver.DeleteTexture)(ctx, ss->Default1D);
547 if (ss->Default2D)
548 (*ctx->Driver.DeleteTexture)(ctx, ss->Default2D);
549 if (ss->Default3D)
550 (*ctx->Driver.DeleteTexture)(ctx, ss->Default3D);
551 if (ss->DefaultCubeMap)
552 (*ctx->Driver.DeleteTexture)(ctx, ss->DefaultCubeMap);
553 if (ss->DefaultRect)
554 (*ctx->Driver.DeleteTexture)(ctx, ss->DefaultRect);
Brian6f472502007-08-13 11:09:48 +0100555 if (ss->Default1DArray)
556 (*ctx->Driver.DeleteTexture)(ctx, ss->Default1DArray);
557 if (ss->Default2DArray)
558 (*ctx->Driver.DeleteTexture)(ctx, ss->Default2DArray);
Brian Paula3f13702003-04-01 16:41:50 +0000559 if (ss)
560 _mesa_free(ss);
561 return GL_FALSE;
jtgafb833d1999-08-19 00:55:39 +0000562}
563
Brian Paulc7e164f2006-06-30 15:44:30 +0000564
Keith Whitwell6dc85572003-07-17 13:43:59 +0000565/**
Brian Paulc7e164f2006-06-30 15:44:30 +0000566 * Callback for deleting a display list. Called by _mesa_HashDeleteAll().
567 */
568static void
569delete_displaylist_cb(GLuint id, void *data, void *userData)
570{
571 struct mesa_display_list *list = (struct mesa_display_list *) data;
572 GLcontext *ctx = (GLcontext *) userData;
573 _mesa_delete_list(ctx, list);
574}
575
576/**
577 * Callback for deleting a texture object. Called by _mesa_HashDeleteAll().
578 */
579static void
580delete_texture_cb(GLuint id, void *data, void *userData)
581{
582 struct gl_texture_object *texObj = (struct gl_texture_object *) data;
583 GLcontext *ctx = (GLcontext *) userData;
584 ctx->Driver.DeleteTexture(ctx, texObj);
585}
586
587/**
588 * Callback for deleting a program object. Called by _mesa_HashDeleteAll().
589 */
590static void
591delete_program_cb(GLuint id, void *data, void *userData)
592{
Brian Paul122629f2006-07-20 16:49:57 +0000593 struct gl_program *prog = (struct gl_program *) data;
Brian Paulc7e164f2006-06-30 15:44:30 +0000594 GLcontext *ctx = (GLcontext *) userData;
595 ctx->Driver.DeleteProgram(ctx, prog);
596}
597
598/**
599 * Callback for deleting an ATI fragment shader object.
600 * Called by _mesa_HashDeleteAll().
601 */
602static void
603delete_fragshader_cb(GLuint id, void *data, void *userData)
604{
605 struct ati_fragment_shader *shader = (struct ati_fragment_shader *) data;
606 GLcontext *ctx = (GLcontext *) userData;
607 _mesa_delete_ati_fragment_shader(ctx, shader);
608}
609
610/**
611 * Callback for deleting a buffer object. Called by _mesa_HashDeleteAll().
612 */
613static void
614delete_bufferobj_cb(GLuint id, void *data, void *userData)
615{
616 struct gl_buffer_object *bufObj = (struct gl_buffer_object *) data;
617 GLcontext *ctx = (GLcontext *) userData;
618 ctx->Driver.DeleteBuffer(ctx, bufObj);
619}
620
Brian Paulc04bb512006-07-11 21:56:43 +0000621/**
622 * Callback for deleting an array object. Called by _mesa_HashDeleteAll().
623 */
624static void
625delete_arrayobj_cb(GLuint id, void *data, void *userData)
626{
627 struct gl_array_object *arrayObj = (struct gl_array_object *) data;
628 GLcontext *ctx = (GLcontext *) userData;
629 _mesa_delete_array_object(ctx, arrayObj);
630}
631
Brian Paul4d4373b2006-11-18 17:44:28 +0000632/**
Brian9e4bae92006-12-20 09:27:42 -0700633 * Callback for deleting shader and shader programs objects.
634 * Called by _mesa_HashDeleteAll().
Brian Paul4d4373b2006-11-18 17:44:28 +0000635 */
636static void
Brian9e4bae92006-12-20 09:27:42 -0700637delete_shader_cb(GLuint id, void *data, void *userData)
Brian Paul4d4373b2006-11-18 17:44:28 +0000638{
Brian9e4bae92006-12-20 09:27:42 -0700639 GLcontext *ctx = (GLcontext *) userData;
640 struct gl_shader *sh = (struct gl_shader *) data;
641 if (sh->Type == GL_FRAGMENT_SHADER || sh->Type == GL_VERTEX_SHADER) {
642 _mesa_free_shader(ctx, sh);
643 }
644 else {
645 struct gl_shader_program *shProg = (struct gl_shader_program *) data;
Brianf3e8c322007-04-18 14:53:23 -0600646 ASSERT(shProg->Type == GL_SHADER_PROGRAM_MESA);
Brian9e4bae92006-12-20 09:27:42 -0700647 _mesa_free_shader_program(ctx, shProg);
648 }
Brian Paul4d4373b2006-11-18 17:44:28 +0000649}
650
Brian Paulc7e164f2006-06-30 15:44:30 +0000651
652/**
653 * Deallocate a shared state object and all children structures.
Keith Whitwell6dc85572003-07-17 13:43:59 +0000654 *
655 * \param ctx GL context.
656 * \param ss shared state pointer.
657 *
658 * Frees the display lists, the texture objects (calling the driver texture
659 * deletion callback to free its private data) and the vertex programs, as well
660 * as their hash tables.
661 *
662 * \sa alloc_shared_state().
jtgafb833d1999-08-19 00:55:39 +0000663 */
Brian Paul178a1c52000-04-22 01:05:00 +0000664static void
665free_shared_state( GLcontext *ctx, struct gl_shared_state *ss )
jtgafb833d1999-08-19 00:55:39 +0000666{
Brian Paulc7e164f2006-06-30 15:44:30 +0000667 /*
668 * Free display lists
669 */
670 _mesa_HashDeleteAll(ss->DisplayList, delete_displaylist_cb, ctx);
Brian Paulbb797902000-01-24 16:19:54 +0000671 _mesa_DeleteHashTable(ss->DisplayList);
jtgafb833d1999-08-19 00:55:39 +0000672
Brian Paulc7e164f2006-06-30 15:44:30 +0000673 /*
674 * Free texture objects
675 */
Brian Paula3f13702003-04-01 16:41:50 +0000676 ASSERT(ctx->Driver.DeleteTexture);
Brian Paul21841f02004-08-14 14:28:11 +0000677 /* the default textures */
Brian Paulc7e164f2006-06-30 15:44:30 +0000678 ctx->Driver.DeleteTexture(ctx, ss->Default1D);
679 ctx->Driver.DeleteTexture(ctx, ss->Default2D);
680 ctx->Driver.DeleteTexture(ctx, ss->Default3D);
681 ctx->Driver.DeleteTexture(ctx, ss->DefaultCubeMap);
682 ctx->Driver.DeleteTexture(ctx, ss->DefaultRect);
Brian6f472502007-08-13 11:09:48 +0100683 ctx->Driver.DeleteTexture(ctx, ss->Default1DArray);
684 ctx->Driver.DeleteTexture(ctx, ss->Default2DArray);
685
Brian Paul21841f02004-08-14 14:28:11 +0000686 /* all other textures */
Brian Paulc7e164f2006-06-30 15:44:30 +0000687 _mesa_HashDeleteAll(ss->TexObjects, delete_texture_cb, ctx);
Brian Paulbb797902000-01-24 16:19:54 +0000688 _mesa_DeleteHashTable(ss->TexObjects);
jtgafb833d1999-08-19 00:55:39 +0000689
Brian Paul1096eae2006-01-16 16:35:13 +0000690#if defined(FEATURE_NV_vertex_program) || defined(FEATURE_NV_fragment_program)
Brian Paulc7e164f2006-06-30 15:44:30 +0000691 _mesa_HashDeleteAll(ss->Programs, delete_program_cb, ctx);
Brian Paul610d5992003-01-14 04:55:45 +0000692 _mesa_DeleteHashTable(ss->Programs);
Brian Paul8dfc5b92002-10-16 17:57:51 +0000693#endif
Brian Paul21841f02004-08-14 14:28:11 +0000694#if FEATURE_ARB_vertex_program
695 _mesa_delete_program(ctx, ss->DefaultVertexProgram);
696#endif
697#if FEATURE_ARB_fragment_program
698 _mesa_delete_program(ctx, ss->DefaultFragmentProgram);
699#endif
Brian Paul1096eae2006-01-16 16:35:13 +0000700
Dave Airlie7f752fe2004-12-19 03:06:59 +0000701#if FEATURE_ATI_fragment_shader
Brian Paulc7e164f2006-06-30 15:44:30 +0000702 _mesa_HashDeleteAll(ss->ATIShaders, delete_fragshader_cb, ctx);
Brian Paul1096eae2006-01-16 16:35:13 +0000703 _mesa_DeleteHashTable(ss->ATIShaders);
704 _mesa_delete_ati_fragment_shader(ctx, ss->DefaultFragmentShader);
Dave Airlie7f752fe2004-12-19 03:06:59 +0000705#endif
Brian Paul30f51ae2001-12-18 04:06:44 +0000706
Tilman Sauerbeck17b50632006-07-11 19:03:21 +0000707#if FEATURE_ARB_vertex_buffer_object || FEATURE_ARB_pixel_buffer_object
Brian Paulc7e164f2006-06-30 15:44:30 +0000708 _mesa_HashDeleteAll(ss->BufferObjects, delete_bufferobj_cb, ctx);
Ian Romanick0207b472003-09-09 00:10:12 +0000709 _mesa_DeleteHashTable(ss->BufferObjects);
Brian Paulddc82ee2005-02-05 19:56:45 +0000710#endif
711
Brian Paulc04bb512006-07-11 21:56:43 +0000712 _mesa_HashDeleteAll(ss->ArrayObjects, delete_arrayobj_cb, ctx);
Tilman Sauerbeckc0eb7772006-07-11 19:36:27 +0000713 _mesa_DeleteHashTable(ss->ArrayObjects);
714
Michal Krol365582d2006-08-01 20:07:31 +0000715#if FEATURE_ARB_shader_objects
Brian9e4bae92006-12-20 09:27:42 -0700716 _mesa_HashDeleteAll(ss->ShaderObjects, delete_shader_cb, ctx);
717 _mesa_DeleteHashTable(ss->ShaderObjects);
Michal Krol365582d2006-08-01 20:07:31 +0000718#endif
Brian Paulddc82ee2005-02-05 19:56:45 +0000719
720#if FEATURE_EXT_framebuffer_object
721 _mesa_DeleteHashTable(ss->FrameBuffers);
722 _mesa_DeleteHashTable(ss->RenderBuffers);
723#endif
Michal Krol9b3752c2005-01-13 14:08:47 +0000724
Keith Whitwelle15fd852002-12-12 13:03:15 +0000725 _glthread_DESTROY_MUTEX(ss->Mutex);
726
Brian Paulc7e164f2006-06-30 15:44:30 +0000727 _mesa_free(ss);
jtgafb833d1999-08-19 00:55:39 +0000728}
729
730
Brian Paul4d859f72004-01-23 18:57:05 +0000731/**
732 * Initialize fields of gl_current_attrib (aka ctx->Current.*)
733 */
734static void
Briand881a9c2006-12-20 09:31:07 -0700735_mesa_init_current(GLcontext *ctx)
jtgafb833d1999-08-19 00:55:39 +0000736{
Brian Paul88bf0382004-02-13 15:30:08 +0000737 GLuint i;
jtgafb833d1999-08-19 00:55:39 +0000738
Brian Paul94b30dc2006-04-25 00:53:25 +0000739 /* Init all to (0,0,0,1) */
Keith Whitwell6dc85572003-07-17 13:43:59 +0000740 for (i = 0; i < VERT_ATTRIB_MAX; i++) {
741 ASSIGN_4V( ctx->Current.Attrib[i], 0.0, 0.0, 0.0, 1.0 );
jtgafb833d1999-08-19 00:55:39 +0000742 }
Brian Paul94b30dc2006-04-25 00:53:25 +0000743
744 /* redo special cases: */
Keith Whitwell6dc85572003-07-17 13:43:59 +0000745 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_WEIGHT], 1.0, 0.0, 0.0, 1.0 );
746 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_NORMAL], 0.0, 0.0, 1.0, 1.0 );
747 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_COLOR0], 1.0, 1.0, 1.0, 1.0 );
Brian Paul88bf0382004-02-13 15:30:08 +0000748 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_COLOR1], 0.0, 0.0, 0.0, 1.0 );
Keith Whitwellfd275602006-10-30 20:16:35 +0000749 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_COLOR_INDEX], 1.0, 0.0, 0.0, 1.0 );
750 ASSIGN_4V( ctx->Current.Attrib[VERT_ATTRIB_EDGEFLAG], 1.0, 0.0, 0.0, 1.0 );
jtgafb833d1999-08-19 00:55:39 +0000751}
752
753
Brian Paul4d859f72004-01-23 18:57:05 +0000754/**
Brian Paul05051032005-11-01 04:36:33 +0000755 * Init vertex/fragment program native limits from logical limits.
756 */
757static void
758init_natives(struct gl_program_constants *prog)
759{
760 prog->MaxNativeInstructions = prog->MaxInstructions;
761 prog->MaxNativeAluInstructions = prog->MaxAluInstructions;
762 prog->MaxNativeTexInstructions = prog->MaxTexInstructions;
763 prog->MaxNativeTexIndirections = prog->MaxTexIndirections;
764 prog->MaxNativeAttribs = prog->MaxAttribs;
765 prog->MaxNativeTemps = prog->MaxTemps;
766 prog->MaxNativeAddressRegs = prog->MaxAddressRegs;
767 prog->MaxNativeParameters = prog->MaxParameters;
768}
769
770
771/**
Brian Paul4d859f72004-01-23 18:57:05 +0000772 * Initialize fields of gl_constants (aka ctx->Const.*).
773 * Use defaults from config.h. The device drivers will often override
774 * some of these values (such as number of texture units).
775 */
Keith Whitwell6dc85572003-07-17 13:43:59 +0000776static void
Briand881a9c2006-12-20 09:31:07 -0700777_mesa_init_constants(GLcontext *ctx)
jtgafb833d1999-08-19 00:55:39 +0000778{
Brian Paul4d053dd2000-01-14 04:45:47 +0000779 assert(ctx);
jtgafb833d1999-08-19 00:55:39 +0000780
Brian Paulcd1cefa2001-06-13 14:56:14 +0000781 assert(MAX_TEXTURE_LEVELS >= MAX_3D_TEXTURE_LEVELS);
782 assert(MAX_TEXTURE_LEVELS >= MAX_CUBE_TEXTURE_LEVELS);
783
Brian Paulda238ee2006-04-13 19:21:58 +0000784 assert(MAX_TEXTURE_UNITS >= MAX_TEXTURE_COORD_UNITS);
785 assert(MAX_TEXTURE_UNITS >= MAX_TEXTURE_IMAGE_UNITS);
786
Brian Paul53f82c52004-10-02 16:39:09 +0000787 /* Constants, may be overriden (usually only reduced) by device drivers */
Brian Paul4d053dd2000-01-14 04:45:47 +0000788 ctx->Const.MaxTextureLevels = MAX_TEXTURE_LEVELS;
Brian Paulcd1cefa2001-06-13 14:56:14 +0000789 ctx->Const.Max3DTextureLevels = MAX_3D_TEXTURE_LEVELS;
790 ctx->Const.MaxCubeTextureLevels = MAX_CUBE_TEXTURE_LEVELS;
Brian Paul8afe7de2002-06-15 03:03:06 +0000791 ctx->Const.MaxTextureRectSize = MAX_TEXTURE_RECT_SIZE;
Ian Romanickbb372f12007-05-16 15:34:22 -0700792 ctx->Const.MaxArrayTextureLayers = MAX_ARRAY_TEXTURE_LAYERS;
Brian Paul610d5992003-01-14 04:55:45 +0000793 ctx->Const.MaxTextureCoordUnits = MAX_TEXTURE_COORD_UNITS;
794 ctx->Const.MaxTextureImageUnits = MAX_TEXTURE_IMAGE_UNITS;
Brian Paulda238ee2006-04-13 19:21:58 +0000795 ctx->Const.MaxTextureUnits = MIN2(ctx->Const.MaxTextureCoordUnits,
796 ctx->Const.MaxTextureImageUnits);
Gareth Hughes2c3d34c2001-03-18 08:53:49 +0000797 ctx->Const.MaxTextureMaxAnisotropy = MAX_TEXTURE_MAX_ANISOTROPY;
Brian Paul87c964d2001-11-06 15:53:00 +0000798 ctx->Const.MaxTextureLodBias = MAX_TEXTURE_LOD_BIAS;
Brian Paul4d053dd2000-01-14 04:45:47 +0000799 ctx->Const.MaxArrayLockSize = MAX_ARRAY_LOCK_SIZE;
Brian Paul539cce52000-02-03 19:40:07 +0000800 ctx->Const.SubPixelBits = SUB_PIXEL_BITS;
801 ctx->Const.MinPointSize = MIN_POINT_SIZE;
802 ctx->Const.MaxPointSize = MAX_POINT_SIZE;
803 ctx->Const.MinPointSizeAA = MIN_POINT_SIZE;
804 ctx->Const.MaxPointSizeAA = MAX_POINT_SIZE;
Brian Paulfde5e2c2001-09-15 18:02:49 +0000805 ctx->Const.PointSizeGranularity = (GLfloat) POINT_SIZE_GRANULARITY;
Brian Paul539cce52000-02-03 19:40:07 +0000806 ctx->Const.MinLineWidth = MIN_LINE_WIDTH;
807 ctx->Const.MaxLineWidth = MAX_LINE_WIDTH;
808 ctx->Const.MinLineWidthAA = MIN_LINE_WIDTH;
809 ctx->Const.MaxLineWidthAA = MAX_LINE_WIDTH;
Brian Paulfde5e2c2001-09-15 18:02:49 +0000810 ctx->Const.LineWidthGranularity = (GLfloat) LINE_WIDTH_GRANULARITY;
Brian Paul4bdcfe52000-04-17 17:57:04 +0000811 ctx->Const.MaxColorTableSize = MAX_COLOR_TABLE_SIZE;
Brian Paul82b02f02000-05-07 20:37:40 +0000812 ctx->Const.MaxConvolutionWidth = MAX_CONVOLUTION_WIDTH;
813 ctx->Const.MaxConvolutionHeight = MAX_CONVOLUTION_HEIGHT;
Brian Paula8644322000-11-27 18:22:13 +0000814 ctx->Const.MaxClipPlanes = MAX_CLIP_PLANES;
815 ctx->Const.MaxLights = MAX_LIGHTS;
Ian Romanick882caa12003-05-30 21:37:14 +0000816 ctx->Const.MaxShininess = 128.0;
Brian Paul53f82c52004-10-02 16:39:09 +0000817 ctx->Const.MaxSpotExponent = 128.0;
818 ctx->Const.MaxViewportWidth = MAX_WIDTH;
819 ctx->Const.MaxViewportHeight = MAX_HEIGHT;
Brian Pauld0492cf2003-04-11 01:20:06 +0000820#if FEATURE_ARB_vertex_program
Brian Paul05051032005-11-01 04:36:33 +0000821 ctx->Const.VertexProgram.MaxInstructions = MAX_NV_VERTEX_PROGRAM_INSTRUCTIONS;
822 ctx->Const.VertexProgram.MaxAluInstructions = 0;
823 ctx->Const.VertexProgram.MaxTexInstructions = 0;
824 ctx->Const.VertexProgram.MaxTexIndirections = 0;
825 ctx->Const.VertexProgram.MaxAttribs = MAX_NV_VERTEX_PROGRAM_INPUTS;
Brian64e80882007-04-16 10:36:28 -0600826 ctx->Const.VertexProgram.MaxTemps = MAX_PROGRAM_TEMPS;
Brian Paul05051032005-11-01 04:36:33 +0000827 ctx->Const.VertexProgram.MaxParameters = MAX_NV_VERTEX_PROGRAM_PARAMS;
828 ctx->Const.VertexProgram.MaxLocalParams = MAX_PROGRAM_LOCAL_PARAMS;
Brian64e80882007-04-16 10:36:28 -0600829 ctx->Const.VertexProgram.MaxEnvParams = MAX_PROGRAM_ENV_PARAMS;
Brian Paul05051032005-11-01 04:36:33 +0000830 ctx->Const.VertexProgram.MaxAddressRegs = MAX_VERTEX_PROGRAM_ADDRESS_REGS;
Briana90046f2006-12-15 10:07:26 -0700831 ctx->Const.VertexProgram.MaxUniformComponents = 4 * MAX_UNIFORMS;
Brian Paul05051032005-11-01 04:36:33 +0000832 init_natives(&ctx->Const.VertexProgram);
Brian Pauld0492cf2003-04-11 01:20:06 +0000833#endif
Briana90046f2006-12-15 10:07:26 -0700834
Brian Pauld0492cf2003-04-11 01:20:06 +0000835#if FEATURE_ARB_fragment_program
Brian Paul05051032005-11-01 04:36:33 +0000836 ctx->Const.FragmentProgram.MaxInstructions = MAX_NV_FRAGMENT_PROGRAM_INSTRUCTIONS;
837 ctx->Const.FragmentProgram.MaxAluInstructions = MAX_FRAGMENT_PROGRAM_ALU_INSTRUCTIONS;
838 ctx->Const.FragmentProgram.MaxTexInstructions = MAX_FRAGMENT_PROGRAM_TEX_INSTRUCTIONS;
839 ctx->Const.FragmentProgram.MaxTexIndirections = MAX_FRAGMENT_PROGRAM_TEX_INDIRECTIONS;
840 ctx->Const.FragmentProgram.MaxAttribs = MAX_NV_FRAGMENT_PROGRAM_INPUTS;
Brian64e80882007-04-16 10:36:28 -0600841 ctx->Const.FragmentProgram.MaxTemps = MAX_PROGRAM_TEMPS;
Brian Paul05051032005-11-01 04:36:33 +0000842 ctx->Const.FragmentProgram.MaxParameters = MAX_NV_FRAGMENT_PROGRAM_PARAMS;
843 ctx->Const.FragmentProgram.MaxLocalParams = MAX_PROGRAM_LOCAL_PARAMS;
Brian64e80882007-04-16 10:36:28 -0600844 ctx->Const.FragmentProgram.MaxEnvParams = MAX_PROGRAM_ENV_PARAMS;
Brian Paul05051032005-11-01 04:36:33 +0000845 ctx->Const.FragmentProgram.MaxAddressRegs = MAX_FRAGMENT_PROGRAM_ADDRESS_REGS;
Briana90046f2006-12-15 10:07:26 -0700846 ctx->Const.FragmentProgram.MaxUniformComponents = 4 * MAX_UNIFORMS;
Brian Paul05051032005-11-01 04:36:33 +0000847 init_natives(&ctx->Const.FragmentProgram);
Brian Pauld0492cf2003-04-11 01:20:06 +0000848#endif
Brian Pauledd67742003-04-18 18:02:43 +0000849 ctx->Const.MaxProgramMatrices = MAX_PROGRAM_MATRICES;
850 ctx->Const.MaxProgramMatrixStackDepth = MAX_PROGRAM_MATRIX_STACK_DEPTH;
Brian Pauld0492cf2003-04-11 01:20:06 +0000851
George Sapountzis507167d2006-12-06 06:54:13 +0200852 /* CheckArrayBounds is overriden by drivers/x11 for X server */
Brian Paula2b9bad2003-11-10 19:08:37 +0000853 ctx->Const.CheckArrayBounds = GL_FALSE;
Brian Paula2b9bad2003-11-10 19:08:37 +0000854
Brian Paul05051032005-11-01 04:36:33 +0000855 /* GL_ARB_draw_buffers */
Brian Paul53f82c52004-10-02 16:39:09 +0000856 ctx->Const.MaxDrawBuffers = MAX_DRAW_BUFFERS;
857
Brian Paul3deaa012005-02-07 05:08:24 +0000858 /* GL_OES_read_format */
Ian Romanick33899b72004-10-16 01:16:54 +0000859 ctx->Const.ColorReadFormat = GL_RGBA;
860 ctx->Const.ColorReadType = GL_UNSIGNED_BYTE;
861
Brian Paul3deaa012005-02-07 05:08:24 +0000862#if FEATURE_EXT_framebuffer_object
863 ctx->Const.MaxColorAttachments = MAX_COLOR_ATTACHMENTS;
864 ctx->Const.MaxRenderbufferSize = MAX_WIDTH;
865#endif
866
Brian Paul90fcf6c2006-11-01 00:12:41 +0000867#if FEATURE_ARB_vertex_shader
868 ctx->Const.MaxVertexTextureImageUnits = MAX_VERTEX_TEXTURE_IMAGE_UNITS;
Briana90046f2006-12-15 10:07:26 -0700869 ctx->Const.MaxVarying = MAX_VARYING;
Brian Paul90fcf6c2006-11-01 00:12:41 +0000870#endif
871
Brian Paul53f82c52004-10-02 16:39:09 +0000872 /* sanity checks */
Brian Paulda238ee2006-04-13 19:21:58 +0000873 ASSERT(ctx->Const.MaxTextureUnits == MIN2(ctx->Const.MaxTextureImageUnits,
874 ctx->Const.MaxTextureCoordUnits));
Brian Paul05051032005-11-01 04:36:33 +0000875 ASSERT(ctx->Const.FragmentProgram.MaxLocalParams <= MAX_PROGRAM_LOCAL_PARAMS);
876 ASSERT(ctx->Const.VertexProgram.MaxLocalParams <= MAX_PROGRAM_LOCAL_PARAMS);
Briana90046f2006-12-15 10:07:26 -0700877
878 ASSERT(MAX_NV_FRAGMENT_PROGRAM_TEMPS <= MAX_PROGRAM_TEMPS);
879 ASSERT(MAX_NV_VERTEX_PROGRAM_TEMPS <= MAX_PROGRAM_TEMPS);
880 ASSERT(MAX_NV_VERTEX_PROGRAM_INPUTS <= VERT_ATTRIB_MAX);
881 ASSERT(MAX_NV_VERTEX_PROGRAM_OUTPUTS <= VERT_RESULT_MAX);
Keith Whitwell6dc85572003-07-17 13:43:59 +0000882}
jtgafb833d1999-08-19 00:55:39 +0000883
Brian Paul4d859f72004-01-23 18:57:05 +0000884
Keith Whitwell6dc85572003-07-17 13:43:59 +0000885/**
Brian Paul5e2e96b2006-05-15 15:26:04 +0000886 * Do some sanity checks on the limits/constants for the given context.
887 * Only called the first time a context is bound.
888 */
889static void
890check_context_limits(GLcontext *ctx)
891{
892 /* Many context limits/constants are limited by the size of
893 * internal arrays.
894 */
895 assert(ctx->Const.MaxTextureImageUnits <= MAX_TEXTURE_IMAGE_UNITS);
896 assert(ctx->Const.MaxTextureCoordUnits <= MAX_TEXTURE_COORD_UNITS);
897 assert(ctx->Const.MaxTextureUnits <= MAX_TEXTURE_IMAGE_UNITS);
898 assert(ctx->Const.MaxTextureUnits <= MAX_TEXTURE_COORD_UNITS);
899
900 assert(ctx->Const.MaxViewportWidth <= MAX_WIDTH);
901 assert(ctx->Const.MaxViewportHeight <= MAX_WIDTH);
902
903 /* make sure largest texture image is <= MAX_WIDTH in size */
904 assert((1 << (ctx->Const.MaxTextureLevels -1 )) <= MAX_WIDTH);
905 assert((1 << (ctx->Const.MaxCubeTextureLevels -1 )) <= MAX_WIDTH);
906 assert((1 << (ctx->Const.Max3DTextureLevels -1 )) <= MAX_WIDTH);
907
908 assert(ctx->Const.MaxDrawBuffers <= MAX_DRAW_BUFFERS);
909
910 /* XXX probably add more tests */
911}
912
913
914/**
Keith Whitwell6dc85572003-07-17 13:43:59 +0000915 * Initialize the attribute groups in a GL context.
916 *
917 * \param ctx GL context.
918 *
919 * Initializes all the attributes, calling the respective <tt>init*</tt>
920 * functions for the more complex data structures.
921 */
922static GLboolean
Briand881a9c2006-12-20 09:31:07 -0700923init_attrib_groups(GLcontext *ctx)
Keith Whitwell6dc85572003-07-17 13:43:59 +0000924{
925 assert(ctx);
Brian Paul4d053dd2000-01-14 04:45:47 +0000926
Keith Whitwell6dc85572003-07-17 13:43:59 +0000927 /* Constants */
928 _mesa_init_constants( ctx );
Brian Paul0771d152000-04-07 00:19:41 +0000929
Brian Paul4d053dd2000-01-14 04:45:47 +0000930 /* Extensions */
Brian Paulde4f4602003-07-03 02:15:06 +0000931 _mesa_init_extensions( ctx );
jtgafb833d1999-08-19 00:55:39 +0000932
Keith Whitwell6dc85572003-07-17 13:43:59 +0000933 /* Attribute Groups */
934 _mesa_init_accum( ctx );
935 _mesa_init_attrib( ctx );
Brian Paul148a2842003-09-17 03:40:11 +0000936 _mesa_init_buffer_objects( ctx );
Keith Whitwell6dc85572003-07-17 13:43:59 +0000937 _mesa_init_color( ctx );
Brian Paul05944c02003-07-22 03:51:46 +0000938 _mesa_init_colortables( ctx );
Keith Whitwell6dc85572003-07-17 13:43:59 +0000939 _mesa_init_current( ctx );
940 _mesa_init_depth( ctx );
941 _mesa_init_debug( ctx );
942 _mesa_init_display_list( ctx );
943 _mesa_init_eval( ctx );
944 _mesa_init_feedback( ctx );
945 _mesa_init_fog( ctx );
946 _mesa_init_histogram( ctx );
947 _mesa_init_hint( ctx );
948 _mesa_init_line( ctx );
949 _mesa_init_lighting( ctx );
950 _mesa_init_matrix( ctx );
Brian Paul67742382005-02-26 17:16:12 +0000951 _mesa_init_multisample( ctx );
Keith Whitwell6dc85572003-07-17 13:43:59 +0000952 _mesa_init_pixel( ctx );
953 _mesa_init_point( ctx );
954 _mesa_init_polygon( ctx );
Brian Paul05944c02003-07-22 03:51:46 +0000955 _mesa_init_program( ctx );
Brian Paul4fb99502005-09-02 13:42:49 +0000956 _mesa_init_query( ctx );
Brian Paulddc82ee2005-02-05 19:56:45 +0000957 _mesa_init_rastpos( ctx );
Brian Paul67742382005-02-26 17:16:12 +0000958 _mesa_init_scissor( ctx );
Brian0bf5dbe2006-12-19 18:02:41 -0700959 _mesa_init_shader_state( ctx );
Keith Whitwell6dc85572003-07-17 13:43:59 +0000960 _mesa_init_stencil( ctx );
961 _mesa_init_transform( ctx );
962 _mesa_init_varray( ctx );
963 _mesa_init_viewport( ctx );
jtgafb833d1999-08-19 00:55:39 +0000964
Keith Whitwell6dc85572003-07-17 13:43:59 +0000965 if (!_mesa_init_texture( ctx ))
966 return GL_FALSE;
Brian Paulb17a7222003-06-13 02:37:27 +0000967
Brian Paul8f04c122004-04-27 13:39:20 +0000968 _mesa_init_texture_s3tc( ctx );
969 _mesa_init_texture_fxt1( ctx );
970
Brian Paul4d053dd2000-01-14 04:45:47 +0000971 /* Miscellaneous */
Keith Whitwella96308c2000-10-30 13:31:59 +0000972 ctx->NewState = _NEW_ALL;
Brian Paul4d053dd2000-01-14 04:45:47 +0000973 ctx->ErrorValue = (GLenum) GL_NO_ERROR;
Brian Paulf782b812002-10-04 17:37:45 +0000974 ctx->_Facing = 0;
Brian Paul4d053dd2000-01-14 04:45:47 +0000975
Brian Paula3f13702003-04-01 16:41:50 +0000976 return GL_TRUE;
jtgafb833d1999-08-19 00:55:39 +0000977}
978
979
Brian Paulf44898c2003-07-18 15:44:57 +0000980/**
Brian Paul21f69782004-11-27 05:05:32 +0000981 * This is the default function we plug into all dispatch table slots
982 * This helps prevents a segfault when someone calls a GL function without
983 * first checking if the extension's supported.
984 */
985static int
986generic_nop(void)
987{
Briancf239ce2007-06-11 10:57:01 -0600988 _mesa_warning(NULL, "User called no-op dispatch function (an unsupported extension function?)");
Brian Paul21f69782004-11-27 05:05:32 +0000989 return 0;
990}
991
992
993/**
994 * Allocate and initialize a new dispatch table.
995 */
996static struct _glapi_table *
997alloc_dispatch_table(void)
998{
999 /* Find the larger of Mesa's dispatch table and libGL's dispatch table.
1000 * In practice, this'll be the same for stand-alone Mesa. But for DRI
1001 * Mesa we do this to accomodate different versions of libGL and various
1002 * DRI drivers.
1003 */
1004 GLint numEntries = MAX2(_glapi_get_dispatch_table_size(),
1005 sizeof(struct _glapi_table) / sizeof(_glapi_proc));
1006 struct _glapi_table *table =
1007 (struct _glapi_table *) _mesa_malloc(numEntries * sizeof(_glapi_proc));
1008 if (table) {
1009 _glapi_proc *entry = (_glapi_proc *) table;
Brian Paula760ccf2004-12-03 15:24:34 +00001010 GLint i;
Brian Paul21f69782004-11-27 05:05:32 +00001011 for (i = 0; i < numEntries; i++) {
1012 entry[i] = (_glapi_proc) generic_nop;
1013 }
1014 }
1015 return table;
1016}
1017
1018
1019/**
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001020 * Initialize a GLcontext struct (rendering context).
Keith Whitwell6dc85572003-07-17 13:43:59 +00001021 *
1022 * This includes allocating all the other structs and arrays which hang off of
1023 * the context by pointers.
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001024 * Note that the driver needs to pass in its dd_function_table here since
1025 * we need to at least call driverFunctions->NewTextureObject to create the
1026 * default texture objects.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001027 *
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001028 * Called by _mesa_create_context().
Keith Whitwell6dc85572003-07-17 13:43:59 +00001029 *
1030 * Performs the imports and exports callback tables initialization, and
1031 * miscellaneous one-time initializations. If no shared context is supplied one
1032 * is allocated, and increase its reference count. Setups the GL API dispatch
1033 * tables. Initialize the TNL module. Sets the maximum Z buffer depth.
1034 * Finally queries the \c MESA_DEBUG and \c MESA_VERBOSE environment variables
1035 * for debug flags.
1036 *
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001037 * \param ctx the context to initialize
1038 * \param visual describes the visual attributes for this context
1039 * \param share_list points to context to share textures, display lists,
1040 * etc with, or NULL
1041 * \param driverFunctions table of device driver functions for this context
1042 * to use
1043 * \param driverContext pointer to driver-specific context data
jtgafb833d1999-08-19 00:55:39 +00001044 */
Brian Paul178a1c52000-04-22 01:05:00 +00001045GLboolean
Briand881a9c2006-12-20 09:31:07 -07001046_mesa_initialize_context(GLcontext *ctx,
1047 const GLvisual *visual,
1048 GLcontext *share_list,
1049 const struct dd_function_table *driverFunctions,
1050 void *driverContext)
jtgafb833d1999-08-19 00:55:39 +00001051{
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001052 ASSERT(driverContext);
1053 assert(driverFunctions->NewTextureObject);
Keith Whitwell3e62d3a2005-03-22 14:27:10 +00001054 assert(driverFunctions->FreeTexImageData);
jtgafb833d1999-08-19 00:55:39 +00001055
Brian Paul4753d602002-06-15 02:38:15 +00001056 /* misc one-time initializations */
1057 one_time_init(ctx);
Brian Paul9a33a112002-06-13 04:28:29 +00001058
Brian Paulb1394fa2000-09-26 20:53:53 +00001059 ctx->Visual = *visual;
Brian Paul3f02f901999-11-24 18:48:30 +00001060 ctx->DrawBuffer = NULL;
1061 ctx->ReadBuffer = NULL;
Brian Paule4b23562005-05-04 20:11:35 +00001062 ctx->WinSysDrawBuffer = NULL;
1063 ctx->WinSysReadBuffer = NULL;
Jouk Jansen5e3bc0c2000-11-22 07:32:16 +00001064
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001065 /* Plug in driver functions and context pointer here.
1066 * This is important because when we call alloc_shared_state() below
1067 * we'll call ctx->Driver.NewTextureObject() to create the default
1068 * textures.
Brian Paula3f13702003-04-01 16:41:50 +00001069 */
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001070 ctx->Driver = *driverFunctions;
1071 ctx->DriverCtx = driverContext;
Brian Paula3f13702003-04-01 16:41:50 +00001072
jtgafb833d1999-08-19 00:55:39 +00001073 if (share_list) {
Brian Paul5a2f32b2001-04-25 18:21:05 +00001074 /* share state with another context */
jtgafb833d1999-08-19 00:55:39 +00001075 ctx->Shared = share_list->Shared;
1076 }
1077 else {
Brian Paul5a2f32b2001-04-25 18:21:05 +00001078 /* allocate new, unshared state */
Brian Paula3f13702003-04-01 16:41:50 +00001079 if (!alloc_shared_state( ctx )) {
Brian Paul4d053dd2000-01-14 04:45:47 +00001080 return GL_FALSE;
jtgafb833d1999-08-19 00:55:39 +00001081 }
1082 }
Brian Paul9560f052000-01-31 23:11:39 +00001083 _glthread_LOCK_MUTEX(ctx->Shared->Mutex);
jtgafb833d1999-08-19 00:55:39 +00001084 ctx->Shared->RefCount++;
Brian Paul9560f052000-01-31 23:11:39 +00001085 _glthread_UNLOCK_MUTEX(ctx->Shared->Mutex);
jtgafb833d1999-08-19 00:55:39 +00001086
Keith Whitwell6dc85572003-07-17 13:43:59 +00001087 if (!init_attrib_groups( ctx )) {
jtgafb833d1999-08-19 00:55:39 +00001088 free_shared_state(ctx, ctx->Shared);
Brian Paul4d053dd2000-01-14 04:45:47 +00001089 return GL_FALSE;
jtgafb833d1999-08-19 00:55:39 +00001090 }
jtgafb833d1999-08-19 00:55:39 +00001091
Brian Paul21f69782004-11-27 05:05:32 +00001092 /* setup the API dispatch tables */
1093 ctx->Exec = alloc_dispatch_table();
1094 ctx->Save = alloc_dispatch_table();
Brian Paul3ab6bbe2000-02-12 17:26:15 +00001095 if (!ctx->Exec || !ctx->Save) {
1096 free_shared_state(ctx, ctx->Shared);
Brian Paul3ab6bbe2000-02-12 17:26:15 +00001097 if (ctx->Exec)
Brian Paul21f69782004-11-27 05:05:32 +00001098 _mesa_free(ctx->Exec);
Brian Paul3ab6bbe2000-02-12 17:26:15 +00001099 }
Brian Paul21f69782004-11-27 05:05:32 +00001100 _mesa_init_exec_table(ctx->Exec);
Brian Paul3ab6bbe2000-02-12 17:26:15 +00001101 ctx->CurrentDispatch = ctx->Exec;
Keith Whitwell6dc85572003-07-17 13:43:59 +00001102#if _HAVE_FULL_GL
Brian Paul21f69782004-11-27 05:05:32 +00001103 _mesa_init_dlist_table(ctx->Save);
Keith Whitwellae0eaf92003-11-24 15:23:18 +00001104 _mesa_install_save_vtxfmt( ctx, &ctx->ListState.ListVtxfmt );
Gareth Hughesd8aa0262001-03-11 18:49:11 +00001105 /* Neutral tnl module stuff */
Keith Whitwell6dc85572003-07-17 13:43:59 +00001106 _mesa_init_exec_vtxfmt( ctx );
Gareth Hughesd8aa0262001-03-11 18:49:11 +00001107 ctx->TnlModule.Current = NULL;
1108 ctx->TnlModule.SwapCount = 0;
Keith Whitwell6dc85572003-07-17 13:43:59 +00001109#endif
Brian Paulb6bcae52001-01-23 23:39:36 +00001110
Brian3e45db62007-03-27 09:51:52 -06001111 ctx->FragmentProgram._MaintainTexEnvProgram
1112 = (_mesa_getenv("MESA_TEX_PROG") != NULL);
1113 ctx->FragmentProgram._UseTexEnvProgram = ctx->FragmentProgram._MaintainTexEnvProgram;
1114
Briana90046f2006-12-15 10:07:26 -07001115 ctx->VertexProgram._MaintainTnlProgram
1116 = (_mesa_getenv("MESA_TNL_PROG") != NULL);
Brian3e45db62007-03-27 09:51:52 -06001117 if (ctx->VertexProgram._MaintainTnlProgram) {
Briana90046f2006-12-15 10:07:26 -07001118 /* this is required... */
1119 ctx->FragmentProgram._MaintainTexEnvProgram = GL_TRUE;
Brian3e45db62007-03-27 09:51:52 -06001120 }
Keith Whitwell47b29f52005-05-04 11:44:44 +00001121
Brian Paula96f8892005-09-13 01:19:29 +00001122 ctx->FirstTimeCurrent = GL_TRUE;
1123
Brian Paul4d053dd2000-01-14 04:45:47 +00001124 return GL_TRUE;
jtgafb833d1999-08-19 00:55:39 +00001125}
1126
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001127
Brian Paulde4f4602003-07-03 02:15:06 +00001128/**
Brian Paul4d053dd2000-01-14 04:45:47 +00001129 * Allocate and initialize a GLcontext structure.
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001130 * Note that the driver needs to pass in its dd_function_table here since
1131 * we need to at least call driverFunctions->NewTextureObject to initialize
1132 * the rendering context.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001133 *
1134 * \param visual a GLvisual pointer (we copy the struct contents)
1135 * \param share_list another context to share display lists with or NULL
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001136 * \param driverFunctions points to the dd_function_table into which the
1137 * driver has plugged in all its special functions.
1138 * \param driverCtx points to the device driver's private context state
Keith Whitwell6dc85572003-07-17 13:43:59 +00001139 *
1140 * \return pointer to a new __GLcontextRec or NULL if error.
Brian Paul4d053dd2000-01-14 04:45:47 +00001141 */
Brian Paul178a1c52000-04-22 01:05:00 +00001142GLcontext *
Briand881a9c2006-12-20 09:31:07 -07001143_mesa_create_context(const GLvisual *visual,
1144 GLcontext *share_list,
1145 const struct dd_function_table *driverFunctions,
1146 void *driverContext)
Brian Paul4d053dd2000-01-14 04:45:47 +00001147{
Brian Paul4753d602002-06-15 02:38:15 +00001148 GLcontext *ctx;
1149
1150 ASSERT(visual);
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001151 ASSERT(driverContext);
Brian Paul4753d602002-06-15 02:38:15 +00001152
Brian Paul3c634522002-10-24 23:57:19 +00001153 ctx = (GLcontext *) _mesa_calloc(sizeof(GLcontext));
Brian Paul4753d602002-06-15 02:38:15 +00001154 if (!ctx)
Brian Paul4d053dd2000-01-14 04:45:47 +00001155 return NULL;
Brian Paul4753d602002-06-15 02:38:15 +00001156
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001157 if (_mesa_initialize_context(ctx, visual, share_list,
1158 driverFunctions, driverContext)) {
Brian Paul4d053dd2000-01-14 04:45:47 +00001159 return ctx;
1160 }
1161 else {
Brian Paul3c634522002-10-24 23:57:19 +00001162 _mesa_free(ctx);
Brian Paul4d053dd2000-01-14 04:45:47 +00001163 return NULL;
1164 }
1165}
1166
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001167
Keith Whitwell6dc85572003-07-17 13:43:59 +00001168/**
Brian Paul4d053dd2000-01-14 04:45:47 +00001169 * Free the data associated with the given context.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001170 *
1171 * But doesn't free the GLcontext struct itself.
1172 *
1173 * \sa _mesa_initialize_context() and init_attrib_groups().
Brian Paul4d053dd2000-01-14 04:45:47 +00001174 */
Brian Paul178a1c52000-04-22 01:05:00 +00001175void
Brian Paulb1394fa2000-09-26 20:53:53 +00001176_mesa_free_context_data( GLcontext *ctx )
Brian Paul4d053dd2000-01-14 04:45:47 +00001177{
Brian Paul4d053dd2000-01-14 04:45:47 +00001178 /* if we're destroying the current context, unbind it first */
Brian Paulb1394fa2000-09-26 20:53:53 +00001179 if (ctx == _mesa_get_current_context()) {
Brian Paule4b23562005-05-04 20:11:35 +00001180 _mesa_make_current(NULL, NULL, NULL);
Brian Paul4d053dd2000-01-14 04:45:47 +00001181 }
Briana510bc32007-03-06 10:07:59 -07001182 else {
1183 /* unreference WinSysDraw/Read buffers */
1184 _mesa_unreference_framebuffer(&ctx->WinSysDrawBuffer);
1185 _mesa_unreference_framebuffer(&ctx->WinSysReadBuffer);
1186 _mesa_unreference_framebuffer(&ctx->DrawBuffer);
1187 _mesa_unreference_framebuffer(&ctx->ReadBuffer);
1188 }
Brian Paul4d053dd2000-01-14 04:45:47 +00001189
Keith Whitwell6dc85572003-07-17 13:43:59 +00001190 _mesa_free_lighting_data( ctx );
1191 _mesa_free_eval_data( ctx );
1192 _mesa_free_texture_data( ctx );
1193 _mesa_free_matrix_data( ctx );
1194 _mesa_free_viewport_data( ctx );
Brian Paul05944c02003-07-22 03:51:46 +00001195 _mesa_free_colortables_data( ctx );
Brian Paul21841f02004-08-14 14:28:11 +00001196 _mesa_free_program_data(ctx);
Brian935f93f2007-03-24 16:20:02 -06001197 _mesa_free_shader_state(ctx);
Brian Paul4fb99502005-09-02 13:42:49 +00001198 _mesa_free_query_data(ctx);
Brian Paul21841f02004-08-14 14:28:11 +00001199
1200#if FEATURE_ARB_vertex_buffer_object
1201 _mesa_delete_buffer_object(ctx, ctx->Array.NullBufferObj);
Brian Paul8dfc5b92002-10-16 17:57:51 +00001202#endif
Brian Paulc04bb512006-07-11 21:56:43 +00001203 _mesa_delete_array_object(ctx, ctx->Array.DefaultArrayObj);
Brian Paulfd284452001-07-19 15:54:34 +00001204
Brian Paul65a66f52004-11-27 22:47:59 +00001205 /* free dispatch tables */
1206 _mesa_free(ctx->Exec);
1207 _mesa_free(ctx->Save);
1208
Brian Paul30f51ae2001-12-18 04:06:44 +00001209 /* Shared context state (display lists, textures, etc) */
Brian Paul9560f052000-01-31 23:11:39 +00001210 _glthread_LOCK_MUTEX(ctx->Shared->Mutex);
Brian Paul4d053dd2000-01-14 04:45:47 +00001211 ctx->Shared->RefCount--;
Brian Paul9560f052000-01-31 23:11:39 +00001212 assert(ctx->Shared->RefCount >= 0);
1213 _glthread_UNLOCK_MUTEX(ctx->Shared->Mutex);
1214 if (ctx->Shared->RefCount == 0) {
Brian Paul4d053dd2000-01-14 04:45:47 +00001215 /* free shared state */
1216 free_shared_state( ctx, ctx->Shared );
1217 }
1218
Brian Paul702ca202003-07-18 15:22:16 +00001219 if (ctx->Extensions.String)
Brian Paulc7e164f2006-06-30 15:44:30 +00001220 _mesa_free((void *) ctx->Extensions.String);
Brian Paul4d053dd2000-01-14 04:45:47 +00001221}
1222
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001223
Keith Whitwell6dc85572003-07-17 13:43:59 +00001224/**
Brian Paul4d053dd2000-01-14 04:45:47 +00001225 * Destroy a GLcontext structure.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001226 *
1227 * \param ctx GL context.
1228 *
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001229 * Calls _mesa_free_context_data() and frees the GLcontext structure itself.
jtgafb833d1999-08-19 00:55:39 +00001230 */
Brian Paul178a1c52000-04-22 01:05:00 +00001231void
Brian Paulb1394fa2000-09-26 20:53:53 +00001232_mesa_destroy_context( GLcontext *ctx )
jtgafb833d1999-08-19 00:55:39 +00001233{
1234 if (ctx) {
Brian Paulb1394fa2000-09-26 20:53:53 +00001235 _mesa_free_context_data(ctx);
Brian Paulc7e164f2006-06-30 15:44:30 +00001236 _mesa_free( (void *) ctx );
jtgafb833d1999-08-19 00:55:39 +00001237 }
1238}
1239
Brian Pauld3fd7ba2004-01-20 02:49:27 +00001240
Keith Whitwell6dc85572003-07-17 13:43:59 +00001241#if _HAVE_FULL_GL
1242/**
jtgafb833d1999-08-19 00:55:39 +00001243 * Copy attribute groups from one context to another.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001244 *
1245 * \param src source context
1246 * \param dst destination context
1247 * \param mask bitwise OR of GL_*_BIT flags
1248 *
1249 * According to the bits specified in \p mask, copies the corresponding
Jose Fonseca375457b2004-09-09 22:23:24 +00001250 * attributes from \p src into \p dst. For many of the attributes a simple \c
Keith Whitwell6dc85572003-07-17 13:43:59 +00001251 * memcpy is not enough due to the existence of internal pointers in their data
1252 * structures.
jtgafb833d1999-08-19 00:55:39 +00001253 */
Brian Paul178a1c52000-04-22 01:05:00 +00001254void
Brian Paulb1394fa2000-09-26 20:53:53 +00001255_mesa_copy_context( const GLcontext *src, GLcontext *dst, GLuint mask )
jtgafb833d1999-08-19 00:55:39 +00001256{
1257 if (mask & GL_ACCUM_BUFFER_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001258 /* OK to memcpy */
1259 dst->Accum = src->Accum;
jtgafb833d1999-08-19 00:55:39 +00001260 }
1261 if (mask & GL_COLOR_BUFFER_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001262 /* OK to memcpy */
1263 dst->Color = src->Color;
jtgafb833d1999-08-19 00:55:39 +00001264 }
1265 if (mask & GL_CURRENT_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001266 /* OK to memcpy */
1267 dst->Current = src->Current;
jtgafb833d1999-08-19 00:55:39 +00001268 }
1269 if (mask & GL_DEPTH_BUFFER_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001270 /* OK to memcpy */
1271 dst->Depth = src->Depth;
jtgafb833d1999-08-19 00:55:39 +00001272 }
1273 if (mask & GL_ENABLE_BIT) {
1274 /* no op */
1275 }
1276 if (mask & GL_EVAL_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001277 /* OK to memcpy */
1278 dst->Eval = src->Eval;
jtgafb833d1999-08-19 00:55:39 +00001279 }
1280 if (mask & GL_FOG_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001281 /* OK to memcpy */
1282 dst->Fog = src->Fog;
jtgafb833d1999-08-19 00:55:39 +00001283 }
1284 if (mask & GL_HINT_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001285 /* OK to memcpy */
1286 dst->Hint = src->Hint;
jtgafb833d1999-08-19 00:55:39 +00001287 }
1288 if (mask & GL_LIGHTING_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001289 GLuint i;
1290 /* begin with memcpy */
Brian Paul2aabdc72006-02-24 18:19:11 +00001291 dst->Light = src->Light;
Brian Paul85d81602002-06-17 23:36:31 +00001292 /* fixup linked lists to prevent pointer insanity */
1293 make_empty_list( &(dst->Light.EnabledList) );
1294 for (i = 0; i < MAX_LIGHTS; i++) {
1295 if (dst->Light.Light[i].Enabled) {
1296 insert_at_tail(&(dst->Light.EnabledList), &(dst->Light.Light[i]));
1297 }
1298 }
jtgafb833d1999-08-19 00:55:39 +00001299 }
1300 if (mask & GL_LINE_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001301 /* OK to memcpy */
1302 dst->Line = src->Line;
jtgafb833d1999-08-19 00:55:39 +00001303 }
1304 if (mask & GL_LIST_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001305 /* OK to memcpy */
1306 dst->List = src->List;
jtgafb833d1999-08-19 00:55:39 +00001307 }
1308 if (mask & GL_PIXEL_MODE_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001309 /* OK to memcpy */
1310 dst->Pixel = src->Pixel;
jtgafb833d1999-08-19 00:55:39 +00001311 }
1312 if (mask & GL_POINT_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001313 /* OK to memcpy */
1314 dst->Point = src->Point;
jtgafb833d1999-08-19 00:55:39 +00001315 }
1316 if (mask & GL_POLYGON_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001317 /* OK to memcpy */
1318 dst->Polygon = src->Polygon;
jtgafb833d1999-08-19 00:55:39 +00001319 }
1320 if (mask & GL_POLYGON_STIPPLE_BIT) {
1321 /* Use loop instead of MEMCPY due to problem with Portland Group's
1322 * C compiler. Reported by John Stone.
1323 */
Brian Paul85d81602002-06-17 23:36:31 +00001324 GLuint i;
1325 for (i = 0; i < 32; i++) {
jtgafb833d1999-08-19 00:55:39 +00001326 dst->PolygonStipple[i] = src->PolygonStipple[i];
1327 }
1328 }
1329 if (mask & GL_SCISSOR_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001330 /* OK to memcpy */
1331 dst->Scissor = src->Scissor;
jtgafb833d1999-08-19 00:55:39 +00001332 }
1333 if (mask & GL_STENCIL_BUFFER_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001334 /* OK to memcpy */
1335 dst->Stencil = src->Stencil;
jtgafb833d1999-08-19 00:55:39 +00001336 }
1337 if (mask & GL_TEXTURE_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001338 /* Cannot memcpy because of pointers */
1339 _mesa_copy_texture_state(src, dst);
jtgafb833d1999-08-19 00:55:39 +00001340 }
1341 if (mask & GL_TRANSFORM_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001342 /* OK to memcpy */
1343 dst->Transform = src->Transform;
jtgafb833d1999-08-19 00:55:39 +00001344 }
1345 if (mask & GL_VIEWPORT_BIT) {
Brian Paul85d81602002-06-17 23:36:31 +00001346 /* Cannot use memcpy, because of pointers in GLmatrix _WindowMap */
1347 dst->Viewport.X = src->Viewport.X;
1348 dst->Viewport.Y = src->Viewport.Y;
1349 dst->Viewport.Width = src->Viewport.Width;
1350 dst->Viewport.Height = src->Viewport.Height;
1351 dst->Viewport.Near = src->Viewport.Near;
1352 dst->Viewport.Far = src->Viewport.Far;
1353 _math_matrix_copy(&dst->Viewport._WindowMap, &src->Viewport._WindowMap);
jtgafb833d1999-08-19 00:55:39 +00001354 }
Brian Paul85d81602002-06-17 23:36:31 +00001355
Keith Whitwella96308c2000-10-30 13:31:59 +00001356 /* XXX FIXME: Call callbacks?
1357 */
1358 dst->NewState = _NEW_ALL;
jtgafb833d1999-08-19 00:55:39 +00001359}
Keith Whitwell23caf202000-11-16 21:05:34 +00001360#endif
Keith Whitwell23caf202000-11-16 21:05:34 +00001361
1362
Brian Paulb1d53d92003-06-11 18:48:19 +00001363/**
1364 * Check if the given context can render into the given framebuffer
1365 * by checking visual attributes.
Brian Paulca007cb2006-03-07 03:01:26 +00001366 *
Brianee170f22007-06-08 14:12:27 -06001367 * Most of these tests could go away because Mesa is now pretty flexible
1368 * in terms of mixing rendering contexts with framebuffers. As long
1369 * as RGB vs. CI mode agree, we're probably good.
Brian Paulca007cb2006-03-07 03:01:26 +00001370 *
Brian Paulb1d53d92003-06-11 18:48:19 +00001371 * \return GL_TRUE if compatible, GL_FALSE otherwise.
1372 */
1373static GLboolean
1374check_compatible(const GLcontext *ctx, const GLframebuffer *buffer)
1375{
1376 const GLvisual *ctxvis = &ctx->Visual;
1377 const GLvisual *bufvis = &buffer->Visual;
1378
1379 if (ctxvis == bufvis)
1380 return GL_TRUE;
1381
1382 if (ctxvis->rgbMode != bufvis->rgbMode)
1383 return GL_FALSE;
Brian Pauld75963d2006-03-07 02:57:04 +00001384#if 0
1385 /* disabling this fixes the fgl_glxgears pbuffer demo */
Brian Paulb1d53d92003-06-11 18:48:19 +00001386 if (ctxvis->doubleBufferMode && !bufvis->doubleBufferMode)
1387 return GL_FALSE;
Brian Pauld75963d2006-03-07 02:57:04 +00001388#endif
Brian Paulb1d53d92003-06-11 18:48:19 +00001389 if (ctxvis->stereoMode && !bufvis->stereoMode)
1390 return GL_FALSE;
1391 if (ctxvis->haveAccumBuffer && !bufvis->haveAccumBuffer)
1392 return GL_FALSE;
1393 if (ctxvis->haveDepthBuffer && !bufvis->haveDepthBuffer)
1394 return GL_FALSE;
1395 if (ctxvis->haveStencilBuffer && !bufvis->haveStencilBuffer)
1396 return GL_FALSE;
1397 if (ctxvis->redMask && ctxvis->redMask != bufvis->redMask)
1398 return GL_FALSE;
1399 if (ctxvis->greenMask && ctxvis->greenMask != bufvis->greenMask)
1400 return GL_FALSE;
1401 if (ctxvis->blueMask && ctxvis->blueMask != bufvis->blueMask)
1402 return GL_FALSE;
Brianee170f22007-06-08 14:12:27 -06001403#if 0
1404 /* disabled (see bug 11161) */
Brian Paulb1d53d92003-06-11 18:48:19 +00001405 if (ctxvis->depthBits && ctxvis->depthBits != bufvis->depthBits)
1406 return GL_FALSE;
Brianee170f22007-06-08 14:12:27 -06001407#endif
Brian Paulb1d53d92003-06-11 18:48:19 +00001408 if (ctxvis->stencilBits && ctxvis->stencilBits != bufvis->stencilBits)
1409 return GL_FALSE;
1410
1411 return GL_TRUE;
1412}
1413
1414
Keith Whitwell6dc85572003-07-17 13:43:59 +00001415/**
Brian Paula702bbf2005-09-14 03:11:36 +00001416 * Do one-time initialization for the given framebuffer. Specifically,
1417 * ask the driver for the window's current size and update the framebuffer
1418 * object to match.
1419 * Really, the device driver should totally take care of this.
1420 */
1421static void
1422initialize_framebuffer_size(GLcontext *ctx, GLframebuffer *fb)
1423{
1424 GLuint width, height;
Brian Paul55e42e52006-10-17 17:43:47 +00001425 if (ctx->Driver.GetBufferSize) {
1426 ctx->Driver.GetBufferSize(fb, &width, &height);
1427 if (ctx->Driver.ResizeBuffers)
1428 ctx->Driver.ResizeBuffers(ctx, fb, width, height);
1429 fb->Initialized = GL_TRUE;
1430 }
Brian Paula702bbf2005-09-14 03:11:36 +00001431}
1432
1433
1434/**
1435 * Bind the given context to the given drawBuffer and readBuffer and
1436 * make it the current context for the calling thread.
1437 * We'll render into the drawBuffer and read pixels from the
1438 * readBuffer (i.e. glRead/CopyPixels, glCopyTexImage, etc).
Keith Whitwell6dc85572003-07-17 13:43:59 +00001439 *
Brian Paula702bbf2005-09-14 03:11:36 +00001440 * We check that the context's and framebuffer's visuals are compatible
1441 * and return immediately if they're not.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001442 *
Brian Paula702bbf2005-09-14 03:11:36 +00001443 * \param newCtx the new GL context. If NULL then there will be no current GL
1444 * context.
1445 * \param drawBuffer the drawing framebuffer
1446 * \param readBuffer the reading framebuffer
Brian Paul00037781999-12-17 14:52:35 +00001447 */
Brian Paulb1394fa2000-09-26 20:53:53 +00001448void
Brian Paule4b23562005-05-04 20:11:35 +00001449_mesa_make_current( GLcontext *newCtx, GLframebuffer *drawBuffer,
1450 GLframebuffer *readBuffer )
Brian Paul00037781999-12-17 14:52:35 +00001451{
Briana90046f2006-12-15 10:07:26 -07001452 GET_CURRENT_CONTEXT(oldCtx);
Briana90046f2006-12-15 10:07:26 -07001453
Keith Whitwell5c728372005-05-12 10:22:29 +00001454 if (MESA_VERBOSE & VERBOSE_API)
Brian Paule4b23562005-05-04 20:11:35 +00001455 _mesa_debug(newCtx, "_mesa_make_current()\n");
Brian Paul00037781999-12-17 14:52:35 +00001456
Brian Paulbe3602d2001-02-28 00:27:48 +00001457 /* Check that the context's and framebuffer's visuals are compatible.
Brian Paulbe3602d2001-02-28 00:27:48 +00001458 */
Brian Paulf1038f82006-03-20 15:20:57 +00001459 if (newCtx && drawBuffer && newCtx->WinSysDrawBuffer != drawBuffer) {
Brian Pauld75963d2006-03-07 02:57:04 +00001460 if (!check_compatible(newCtx, drawBuffer)) {
1461 _mesa_warning(newCtx,
1462 "MakeCurrent: incompatible visuals for context and drawbuffer");
Brian Paulb1d53d92003-06-11 18:48:19 +00001463 return;
Brian Pauld75963d2006-03-07 02:57:04 +00001464 }
Brian Paulb1d53d92003-06-11 18:48:19 +00001465 }
Brian Paulf1038f82006-03-20 15:20:57 +00001466 if (newCtx && readBuffer && newCtx->WinSysReadBuffer != readBuffer) {
Brian Pauld75963d2006-03-07 02:57:04 +00001467 if (!check_compatible(newCtx, readBuffer)) {
1468 _mesa_warning(newCtx,
1469 "MakeCurrent: incompatible visuals for context and readbuffer");
Brian Paulb1d53d92003-06-11 18:48:19 +00001470 return;
Brian Pauld75963d2006-03-07 02:57:04 +00001471 }
Brian Paulbe3602d2001-02-28 00:27:48 +00001472 }
1473
Brian Paulc6c0f942006-03-16 18:05:25 +00001474 /* We used to call _glapi_check_multithread() here. Now do it in drivers */
Brian Paulf9b97d92000-01-28 20:17:42 +00001475 _glapi_set_context((void *) newCtx);
Brian Paulb1394fa2000-09-26 20:53:53 +00001476 ASSERT(_mesa_get_current_context() == newCtx);
Keith Whitwell23caf202000-11-16 21:05:34 +00001477
Briane6a93812007-02-26 11:37:37 -07001478 if (oldCtx) {
Briana510bc32007-03-06 10:07:59 -07001479 _mesa_unreference_framebuffer(&oldCtx->WinSysDrawBuffer);
1480 _mesa_unreference_framebuffer(&oldCtx->WinSysReadBuffer);
Brian4fc46a62007-04-14 08:06:54 -06001481 _mesa_unreference_framebuffer(&oldCtx->DrawBuffer);
1482 _mesa_unreference_framebuffer(&oldCtx->ReadBuffer);
Briane6a93812007-02-26 11:37:37 -07001483 }
1484
Keith Whitwell23caf202000-11-16 21:05:34 +00001485 if (!newCtx) {
Brian Paul00037781999-12-17 14:52:35 +00001486 _glapi_set_dispatch(NULL); /* none current */
1487 }
Keith Whitwell23caf202000-11-16 21:05:34 +00001488 else {
1489 _glapi_set_dispatch(newCtx->CurrentDispatch);
Brian Paul00037781999-12-17 14:52:35 +00001490
Keith Whitwell23caf202000-11-16 21:05:34 +00001491 if (drawBuffer && readBuffer) {
1492 /* TODO: check if newCtx and buffer's visual match??? */
Brian Paule4b23562005-05-04 20:11:35 +00001493
Brian Paule4b23562005-05-04 20:11:35 +00001494 ASSERT(drawBuffer->Name == 0);
1495 ASSERT(readBuffer->Name == 0);
Briana510bc32007-03-06 10:07:59 -07001496 _mesa_reference_framebuffer(&newCtx->WinSysDrawBuffer, drawBuffer);
1497 _mesa_reference_framebuffer(&newCtx->WinSysReadBuffer, readBuffer);
Brian Paulf1038f82006-03-20 15:20:57 +00001498
1499 /*
1500 * Only set the context's Draw/ReadBuffer fields if they're NULL
1501 * or not bound to a user-created FBO.
1502 */
Brian Paule4b23562005-05-04 20:11:35 +00001503 if (!newCtx->DrawBuffer || newCtx->DrawBuffer->Name == 0) {
Briana510bc32007-03-06 10:07:59 -07001504 _mesa_reference_framebuffer(&newCtx->DrawBuffer, drawBuffer);
Roland Scheideggercbfe29c2007-07-16 18:21:36 +02001505 /* fix up the fb fields - these will end up wrong otherwise
Roland Scheideggera1bc0d02007-07-18 20:17:14 +02001506 if the DRIdrawable changes, and everything relies on them.
1507 This is a bit messy (same as needed in _mesa_BindFramebufferEXT) */
Roland Scheideggercbfe29c2007-07-16 18:21:36 +02001508 int i;
1509 GLenum buffers[MAX_DRAW_BUFFERS];
1510 for(i = 0; i < newCtx->Const.MaxDrawBuffers; i++) {
1511 buffers[i] = newCtx->Color.DrawBuffer[i];
1512 }
1513 _mesa_drawbuffers(newCtx, newCtx->Const.MaxDrawBuffers, buffers, NULL);
Brian Paulf1038f82006-03-20 15:20:57 +00001514 }
1515 if (!newCtx->ReadBuffer || newCtx->ReadBuffer->Name == 0) {
Briana510bc32007-03-06 10:07:59 -07001516 _mesa_reference_framebuffer(&newCtx->ReadBuffer, readBuffer);
Roland Scheideggera1bc0d02007-07-18 20:17:14 +02001517 _mesa_readbuffer_update_fields(newCtx, newCtx->Pixel.ReadBuffer);
Brian Paule4b23562005-05-04 20:11:35 +00001518 }
Brian Paulbb5c84f2005-07-01 01:22:25 +00001519
Keith Whitwell23caf202000-11-16 21:05:34 +00001520 newCtx->NewState |= _NEW_BUFFERS;
Brian Paul10d7f542002-06-17 23:38:14 +00001521
Brian Paul4d4add02006-10-15 19:26:43 +00001522#if 1
1523 /* We want to get rid of these lines: */
1524
Keith Whitwell6dc85572003-07-17 13:43:59 +00001525#if _HAVE_FULL_GL
Brian Paul65a66f52004-11-27 22:47:59 +00001526 if (!drawBuffer->Initialized) {
Brian Paula702bbf2005-09-14 03:11:36 +00001527 initialize_framebuffer_size(newCtx, drawBuffer);
Brian Paul10d7f542002-06-17 23:38:14 +00001528 }
Brian Paul65a66f52004-11-27 22:47:59 +00001529 if (readBuffer != drawBuffer && !readBuffer->Initialized) {
Brian Paula702bbf2005-09-14 03:11:36 +00001530 initialize_framebuffer_size(newCtx, readBuffer);
Brian Paul10d7f542002-06-17 23:38:14 +00001531 }
Keith Whitwellf9bfdb12006-09-22 11:36:30 +00001532
1533 _mesa_resizebuffers(newCtx);
Keith Whitwell6dc85572003-07-17 13:43:59 +00001534#endif
Brian Paul4d4add02006-10-15 19:26:43 +00001535
1536#else
1537 /* We want the drawBuffer and readBuffer to be initialized by
1538 * the driver.
1539 * This generally means the Width and Height match the actual
1540 * window size and the renderbuffers (both hardware and software
1541 * based) are allocated to match. The later can generally be
1542 * done with a call to _mesa_resize_framebuffer().
1543 *
1544 * It's theoretically possible for a buffer to have zero width
1545 * or height, but for now, assert check that the driver did what's
1546 * expected of it.
1547 */
1548 ASSERT(drawBuffer->Width > 0);
1549 ASSERT(drawBuffer->Height > 0);
1550#endif
1551
Brian Paul65a66f52004-11-27 22:47:59 +00001552 if (newCtx->FirstTimeCurrent) {
1553 /* set initial viewport and scissor size now */
Brian Paula702bbf2005-09-14 03:11:36 +00001554 _mesa_set_viewport(newCtx, 0, 0,
1555 drawBuffer->Width, drawBuffer->Height);
Brian Pauldb79d2a2006-03-29 18:41:19 +00001556 _mesa_set_scissor(newCtx, 0, 0,
1557 drawBuffer->Width, drawBuffer->Height );
Brian Paul5e2e96b2006-05-15 15:26:04 +00001558 check_context_limits(newCtx);
Brian Paul65a66f52004-11-27 22:47:59 +00001559 }
Brian Paul00037781999-12-17 14:52:35 +00001560 }
Keith Whitwell23caf202000-11-16 21:05:34 +00001561
Keith Whitwell23caf202000-11-16 21:05:34 +00001562 /* We can use this to help debug user's problems. Tell them to set
1563 * the MESA_INFO env variable before running their app. Then the
1564 * first time each context is made current we'll print some useful
1565 * information.
1566 */
1567 if (newCtx->FirstTimeCurrent) {
Brian Paul3c634522002-10-24 23:57:19 +00001568 if (_mesa_getenv("MESA_INFO")) {
Keith Whitwell6dc85572003-07-17 13:43:59 +00001569 _mesa_print_info();
Keith Whitwell23caf202000-11-16 21:05:34 +00001570 }
1571 newCtx->FirstTimeCurrent = GL_FALSE;
1572 }
Brian Paul00037781999-12-17 14:52:35 +00001573 }
1574}
1575
Brian Paul635ee2d2005-04-15 17:25:07 +00001576
1577/**
1578 * Make context 'ctx' share the display lists, textures and programs
1579 * that are associated with 'ctxToShare'.
1580 * Any display lists, textures or programs associated with 'ctx' will
1581 * be deleted if nobody else is sharing them.
1582 */
1583GLboolean
1584_mesa_share_state(GLcontext *ctx, GLcontext *ctxToShare)
1585{
1586 if (ctx && ctxToShare && ctx->Shared && ctxToShare->Shared) {
1587 ctx->Shared->RefCount--;
1588 if (ctx->Shared->RefCount == 0) {
1589 free_shared_state(ctx, ctx->Shared);
1590 }
1591 ctx->Shared = ctxToShare->Shared;
1592 ctx->Shared->RefCount++;
1593 return GL_TRUE;
1594 }
1595 else {
1596 return GL_FALSE;
1597 }
1598}
1599
1600
1601
Keith Whitwell6dc85572003-07-17 13:43:59 +00001602/**
Briand881a9c2006-12-20 09:31:07 -07001603 * \return pointer to the current GL context for this thread.
Keith Whitwell6dc85572003-07-17 13:43:59 +00001604 *
1605 * Calls _glapi_get_context(). This isn't the fastest way to get the current
Briand881a9c2006-12-20 09:31:07 -07001606 * context. If you need speed, see the #GET_CURRENT_CONTEXT macro in
1607 * context.h.
Brian Paul00037781999-12-17 14:52:35 +00001608 */
Brian Paulb1394fa2000-09-26 20:53:53 +00001609GLcontext *
1610_mesa_get_current_context( void )
Brian Paul00037781999-12-17 14:52:35 +00001611{
Brian Paulf9b97d92000-01-28 20:17:42 +00001612 return (GLcontext *) _glapi_get_context();
Brian Paul00037781999-12-17 14:52:35 +00001613}
1614
Briand881a9c2006-12-20 09:31:07 -07001615
Keith Whitwell6dc85572003-07-17 13:43:59 +00001616/**
1617 * Get context's current API dispatch table.
1618 *
1619 * It'll either be the immediate-mode execute dispatcher or the display list
1620 * compile dispatcher.
1621 *
1622 * \param ctx GL context.
1623 *
1624 * \return pointer to dispatch_table.
1625 *
1626 * Simply returns __GLcontextRec::CurrentDispatch.
Brian Paulfbd8f211999-11-11 01:22:25 +00001627 */
1628struct _glapi_table *
1629_mesa_get_dispatch(GLcontext *ctx)
1630{
1631 return ctx->CurrentDispatch;
1632}
1633
Keith Whitwell6dc85572003-07-17 13:43:59 +00001634/*@}*/
Brian Paulfbd8f211999-11-11 01:22:25 +00001635
1636
jtgafb833d1999-08-19 00:55:39 +00001637/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +00001638/** \name Miscellaneous functions */
jtgafb833d1999-08-19 00:55:39 +00001639/**********************************************************************/
Keith Whitwell6dc85572003-07-17 13:43:59 +00001640/*@{*/
jtgafb833d1999-08-19 00:55:39 +00001641
Keith Whitwell6dc85572003-07-17 13:43:59 +00001642/**
1643 * Record an error.
1644 *
1645 * \param ctx GL context.
1646 * \param error error code.
1647 *
1648 * Records the given error code and call the driver's dd_function_table::Error
1649 * function if defined.
1650 *
1651 * \sa
Brian Paul4e9676f2002-06-29 19:48:15 +00001652 * This is called via _mesa_error().
jtgafb833d1999-08-19 00:55:39 +00001653 */
Brian Paulb1394fa2000-09-26 20:53:53 +00001654void
Briand881a9c2006-12-20 09:31:07 -07001655_mesa_record_error(GLcontext *ctx, GLenum error)
jtgafb833d1999-08-19 00:55:39 +00001656{
Brian Paul18a285a2002-03-16 00:53:15 +00001657 if (!ctx)
1658 return;
1659
Brian Paul7eb06032000-07-14 04:13:40 +00001660 if (ctx->ErrorValue == GL_NO_ERROR) {
jtgafb833d1999-08-19 00:55:39 +00001661 ctx->ErrorValue = error;
1662 }
1663
1664 /* Call device driver's error handler, if any. This is used on the Mac. */
1665 if (ctx->Driver.Error) {
Briand881a9c2006-12-20 09:31:07 -07001666 ctx->Driver.Error(ctx);
jtgafb833d1999-08-19 00:55:39 +00001667 }
1668}
1669
Briand881a9c2006-12-20 09:31:07 -07001670
Keith Whitwell6dc85572003-07-17 13:43:59 +00001671/**
1672 * Execute glFinish().
1673 *
1674 * Calls the #ASSERT_OUTSIDE_BEGIN_END_AND_FLUSH macro and the
1675 * dd_function_table::Finish driver callback, if not NULL.
1676 */
Kendall Bennettc40d1dd2003-10-21 22:22:17 +00001677void GLAPIENTRY
Briand881a9c2006-12-20 09:31:07 -07001678_mesa_Finish(void)
jtgafb833d1999-08-19 00:55:39 +00001679{
Brian Paulfa9df402000-02-02 19:16:46 +00001680 GET_CURRENT_CONTEXT(ctx);
Keith Whitwellcab974c2000-12-26 05:09:27 +00001681 ASSERT_OUTSIDE_BEGIN_END_AND_FLUSH(ctx);
Brian Paulfa9df402000-02-02 19:16:46 +00001682 if (ctx->Driver.Finish) {
Briand881a9c2006-12-20 09:31:07 -07001683 ctx->Driver.Finish(ctx);
jtgafb833d1999-08-19 00:55:39 +00001684 }
1685}
1686
Briand881a9c2006-12-20 09:31:07 -07001687
Keith Whitwell6dc85572003-07-17 13:43:59 +00001688/**
1689 * Execute glFlush().
1690 *
1691 * Calls the #ASSERT_OUTSIDE_BEGIN_END_AND_FLUSH macro and the
1692 * dd_function_table::Flush driver callback, if not NULL.
1693 */
Kendall Bennettc40d1dd2003-10-21 22:22:17 +00001694void GLAPIENTRY
Briand881a9c2006-12-20 09:31:07 -07001695_mesa_Flush(void)
jtgafb833d1999-08-19 00:55:39 +00001696{
Brian Paulfa9df402000-02-02 19:16:46 +00001697 GET_CURRENT_CONTEXT(ctx);
Keith Whitwellcab974c2000-12-26 05:09:27 +00001698 ASSERT_OUTSIDE_BEGIN_END_AND_FLUSH(ctx);
Brian Paulfa9df402000-02-02 19:16:46 +00001699 if (ctx->Driver.Flush) {
Briand881a9c2006-12-20 09:31:07 -07001700 ctx->Driver.Flush(ctx);
jtgafb833d1999-08-19 00:55:39 +00001701 }
jtgafb833d1999-08-19 00:55:39 +00001702}
Brian Paul48c6a6e2000-09-08 21:28:04 +00001703
1704
Keith Whitwell6dc85572003-07-17 13:43:59 +00001705/*@}*/