123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371 |
- //-----------------------------------------------------------------------------
- // Copyright (c) 2013 GarageGames, LLC
- // Portions Copyright (c) 2014 James S Urquhart
- //
- // Permission is hereby granted, free of charge, to any person obtaining a copy
- // of this software and associated documentation files (the "Software"), to
- // deal in the Software without restriction, including without limitation the
- // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
- // sell copies of the Software, and to permit persons to whom the Software is
- // furnished to do so, subject to the following conditions:
- //
- // The above copyright notice and this permission notice shall be included in
- // all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
- // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
- // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
- // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
- // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
- // IN THE SOFTWARE.
- //-----------------------------------------------------------------------------
- // Many OSX frameworks include OpenTransport, and OT's new operator conflicts
- // with our redefinition of 'new', so we have to pre-include platformAndroid.h,
- // which contains the workaround.
- #include "platformEmscripten/platformEmscripten.h"
- #include "platformEmscripten/platformGL.h"
- #include "console/console.h"
- #include "graphics/dgl.h"
- #define TEST_FOR_OPENGL_ERRORS 0;
- GLState gGLState;
- bool gOpenGLDisablePT = false;
- bool gOpenGLDisableCVA = false;
- bool gOpenGLDisableTEC = false;
- bool gOpenGLDisableARBMT = false;
- bool gOpenGLDisableFC = true; //false;
- bool gOpenGLDisableTCompress = false;
- bool gOpenGLNoEnvColor = false;
- float gOpenGLGammaCorrection = 0.5;
- bool gOpenGLNoDrawArraysAlpha = false;
- // Find out which extensions are available for this renderer.
- void getGLCapabilities( )
- {
- AssertFatal(platState.engine, "getGLCapabilities() was called before a monitor was chosen!");
- // silently create an opengl context on the current display,
- // so that we can get valid renderer and capability info.
- // we save off the current context so that we can silently restore it.
- // the user should not be aware of this little shuffle.
- //CGLContextObj curr_ctx = CGLGetCurrentContext ();
- //CGLContextObj temp_ctx = getContextForCapsCheck();
- /*
- if(!temp_ctx)
- {
- Con::errorf("OpenGL may not be set up correctly!");
- return;
- }
- */
- //CGLSetCurrentContext(temp_ctx);
- // Get the OpenGL info strings we'll need
- const char* pVendString = (const char*) glGetString( GL_VENDOR );
- const char* pRendString = (const char*) glGetString( GL_RENDERER );
- const char* pVersString = (const char*) glGetString( GL_VERSION );
- const char* pExtString = (const char*) glGetString( GL_EXTENSIONS );
- // Output some driver info to the console:
- Con::printf( "OpenGL driver information:" );
- if ( pVendString )
- Con::printf( " Vendor: %s", pVendString );
- if ( pRendString )
- Con::printf( " Renderer: %s", pRendString );
- if ( pVersString )
- Con::printf( " Version: %s", pVersString );
- // pre-clear the structure
- dMemset(&gGLState, 0, sizeof(gGLState));
- if(pExtString)
- {
- // EXT_paletted_texture ========================================
- if (dStrstr(pExtString, (const char*)"GL_EXT_paletted_texture") != NULL)
- gGLState.suppPalettedTexture = true;
-
- // EXT_compiled_vertex_array ========================================
- gGLState.suppLockedArrays = false;
- if (dStrstr(pExtString, (const char*)"GL_EXT_compiled_vertex_array") != NULL)
- gGLState.suppLockedArrays = true;
- // ARB_multitexture ========================================
- if (dStrstr(pExtString, (const char*)"GL_ARB_multitexture") != NULL)
- gGLState.suppARBMultitexture = true;
-
- // EXT_blend_color
- if(dStrstr(pExtString, (const char*)"GL_EXT_blend_color") != NULL)
- gGLState.suppEXTblendcolor = true;
- // EXT_blend_minmax
- if(dStrstr(pExtString, (const char*)"GL_EXT_blend_minmax") != NULL)
- gGLState.suppEXTblendminmax = true;
- // NV_vertex_array_range ========================================
- // does not appear to be supported by apple, at all. ( as of 10.4.3 )
- // GL_APPLE_vertex_array_range is similar, and may be nearly identical.
- if (dStrstr(pExtString, (const char*)"GL_NV_vertex_array_range") != NULL)
- gGLState.suppVertexArrayRange = true;
-
- // EXT_fog_coord ========================================
- if (dStrstr(pExtString, (const char*)"GL_EXT_fog_coord") != NULL)
- gGLState.suppFogCoord = true;
-
- // ARB_texture_compression ========================================
- if (dStrstr(pExtString, (const char*)"GL_ARB_texture_compression") != NULL)
- gGLState.suppTextureCompression = true;
- // 3DFX_texture_compression_FXT1 ========================================
- if (dStrstr(pExtString, (const char*)"GL_3DFX_texture_compression_FXT1") != NULL)
- gGLState.suppFXT1 = true;
- // EXT_texture_compression_S3TC ========================================
- if (dStrstr(pExtString, (const char*)"GL_EXT_texture_compression_s3tc") != NULL)
- gGLState.suppS3TC = true;
- // EXT_vertex_buffer ========================================
- // This extension is deprecated, and not supported by Apple. ( 10.4.3 )
- // Instead, the ARB Vertex Buffer extension is supported.
- // The new extension has a different API, so TGE should be updated to use it.
- if (dStrstr(pExtString, (const char*)"GL_EXT_vertex_buffer") != NULL)
- gGLState.suppVertexBuffer = true;
- // Anisotropic filtering ========================================
- gGLState.suppTexAnisotropic = (dStrstr(pExtString, (const char*)"GL_EXT_texture_filter_anisotropic") != NULL);
- if (gGLState.suppTexAnisotropic)
- glGetFloatv(GL_MAX_TEXTURE_MAX_ANISOTROPY_EXT, &gGLState.maxAnisotropy);
- // Binary states, i.e., no supporting functions ========================================
- // NOTE:
- // Some of these have multiple representations, via EXT and|or ARB and|or NV and|or SGIS ... etc.
- // Check all relative versions.
-
- gGLState.suppPackedPixels = (dStrstr(pExtString, (const char*)"GL_EXT_packed_pixels") != NULL);
- gGLState.suppPackedPixels |= (dStrstr(pExtString, (const char*)"GL_APPLE_packed_pixel") != NULL);
- gGLState.suppTextureEnvCombine = (dStrstr(pExtString, (const char*)"GL_EXT_texture_env_combine") != NULL);
- gGLState.suppTextureEnvCombine|= (dStrstr(pExtString, (const char*)"GL_ARB_texture_env_combine") != NULL);
- gGLState.suppEdgeClamp = (dStrstr(pExtString, (const char*)"GL_EXT_texture_edge_clamp") != NULL);
- gGLState.suppEdgeClamp |= (dStrstr(pExtString, (const char*)"GL_SGIS_texture_edge_clamp") != NULL);
- gGLState.suppEdgeClamp |= (dStrstr(pExtString, (const char*)"GL_ARB_texture_border_clamp") != NULL);
- gGLState.suppEdgeClamp = true;
- gGLState.suppTexEnvAdd = (dStrstr(pExtString, (const char*)"GL_ARB_texture_env_add") != NULL);
- gGLState.suppTexEnvAdd |= (dStrstr(pExtString, (const char*)"GL_EXT_texture_env_add") != NULL);
- }
-
- // Texture combine units ========================================
- // if (gGLState.suppARBMultitexture)
- // glGetIntegerv(GL_MAX_TEXTURE_UNITS_ARB, &gGLState.maxTextureUnits);
- gGLState.suppARBMultitexture = false;
- gGLState.maxTextureUnits = 1;
- // Swap interval ========================================
- // Mac inherently supports a swap interval via AGL-set-integer.
- gGLState.suppSwapInterval = true;
- // FSAA support, TODO: check for ARB multisample support
- // multisample support should be checked via CGL
- gGLState.maxFSAASamples = 4;
- // dump found extensions to the console...
- Con::printf("OpenGL Init: Enabled Extensions");
- if (gGLState.suppARBMultitexture) Con::printf(" ARB_multitexture (Max Texture Units: %d)", gGLState.maxTextureUnits);
- if (gGLState.suppEXTblendcolor) Con::printf(" EXT_blend_color");
- if (gGLState.suppEXTblendminmax) Con::printf(" EXT_blend_minmax");
- if (gGLState.suppPalettedTexture) Con::printf(" EXT_paletted_texture");
- if (gGLState.suppLockedArrays) Con::printf(" EXT_compiled_vertex_array");
- if (gGLState.suppVertexArrayRange) Con::printf(" NV_vertex_array_range");
- if (gGLState.suppTextureEnvCombine) Con::printf(" EXT_texture_env_combine");
- if (gGLState.suppPackedPixels) Con::printf(" EXT_packed_pixels");
- if (gGLState.suppFogCoord) Con::printf(" EXT_fog_coord");
- if (gGLState.suppTextureCompression) Con::printf(" ARB_texture_compression");
- if (gGLState.suppS3TC) Con::printf(" EXT_texture_compression_s3tc");
- if (gGLState.suppFXT1) Con::printf(" 3DFX_texture_compression_FXT1");
- if (gGLState.suppTexEnvAdd) Con::printf(" (ARB|EXT)_texture_env_add");
- if (gGLState.suppTexAnisotropic) Con::printf(" EXT_texture_filter_anisotropic (Max anisotropy: %f)", gGLState.maxAnisotropy);
- if (gGLState.suppSwapInterval) Con::printf(" Vertical Sync");
- if (gGLState.maxFSAASamples) Con::printf(" ATI_FSAA");
- Con::warnf("OpenGL Init: Disabled Extensions");
- if (!gGLState.suppARBMultitexture) Con::warnf(" ARB_multitexture");
- if (!gGLState.suppEXTblendcolor) Con::warnf(" EXT_blend_color");
- if (!gGLState.suppEXTblendminmax) Con::warnf(" EXT_blend_minmax");
- if (!gGLState.suppPalettedTexture) Con::warnf(" EXT_paletted_texture");
- if (!gGLState.suppLockedArrays) Con::warnf(" EXT_compiled_vertex_array");
- if (!gGLState.suppVertexArrayRange) Con::warnf(" NV_vertex_array_range");
- if (!gGLState.suppTextureEnvCombine) Con::warnf(" EXT_texture_env_combine");
- if (!gGLState.suppPackedPixels) Con::warnf(" EXT_packed_pixels");
- if (!gGLState.suppFogCoord) Con::warnf(" EXT_fog_coord");
- if (!gGLState.suppTextureCompression) Con::warnf(" ARB_texture_compression");
- if (!gGLState.suppS3TC) Con::warnf(" EXT_texture_compression_s3tc");
- if (!gGLState.suppFXT1) Con::warnf(" 3DFX_texture_compression_FXT1");
- if (!gGLState.suppTexEnvAdd) Con::warnf(" (ARB|EXT)_texture_env_add");
- if (!gGLState.suppTexAnisotropic) Con::warnf(" EXT_texture_filter_anisotropic");
- if (!gGLState.suppSwapInterval) Con::warnf(" Vertical Sync");
- if (!gGLState.maxFSAASamples) Con::warnf(" ATI_FSAA");
- Con::printf("");
- // Set some console variables:
- Con::setBoolVariable( "$FogCoordSupported", gGLState.suppFogCoord );
- Con::setBoolVariable( "$TextureCompressionSupported", gGLState.suppTextureCompression );
- Con::setBoolVariable( "$AnisotropySupported", gGLState.suppTexAnisotropic );
- Con::setBoolVariable( "$PalettedTextureSupported", gGLState.suppPalettedTexture );
- Con::setBoolVariable( "$SwapIntervalSupported", gGLState.suppSwapInterval );
- if (!gGLState.suppPalettedTexture && Con::getBoolVariable("$pref::OpenGL::forcePalettedTexture",false))
- {
- Con::setBoolVariable("$pref::OpenGL::forcePalettedTexture", false);
- Con::setBoolVariable("$pref::OpenGL::force16BitTexture", true);
- }
- // get fsaa samples. default to normal, no antialiasing
- // TODO: clamp this against ARB_multisample capabilities.
- gFSAASamples = Con::getIntVariable("$pref::OpenGL::numFSAASamples", 1);
- }
- void dglSetFSAASamples(GLint aasamp)
- {
- if (gGLState.maxFSAASamples < 2)
- return;
-
- // fix out of range values
- if (aasamp < 1)
- aasamp = 1;
- else if (aasamp == 3)
- aasamp = 2;
- else if (aasamp > 4)
- aasamp = 4;
-
- /*
- THIS IS CARBON
- aglSetInteger(platState.ctx, ATI_FSAA_LEVEL, &aasamp);
-
- IT NEEDS OT BE REPLACED WITH SOMETHING LIKE THIS:
- const char *glRenderer = (const char *) glGetString(GL_RENDERER);
-
- if (strstr(glRenderer, "ATI"))
- {
- NSOpenGLContext* testContext; /// <<< Need to get ahold of the current context
- [testContext setValues:&aasamp forParameter:ATI_FSAA_LEVEL];
- }
- */
-
- Con::printf(">>Number of FSAA samples is now [%d].", aasamp);
- Con::setIntVariable("$pref::OpenGL::numFSAASamples", aasamp);
- }
- #ifdef DUMMY_GL
- // declare stub functions
- #define GL_FUNCTION(fn_return, fn_name, fn_args, fn_value) fn_return fn_name fn_args{ printf(" fn_name \n"); fn_value }
- #include "platform/GLCoreFunc.h"
- #include "platform/GLExtFunc.h"
- #undef GL_FUNCTION
- // point gl function pointers at stub functions
- /*#define GL_FUNCTION(fn_return,fn_name,fn_args, fn_value) fn_return (*fn_name)fn_args = stub_##fn_name;
- #include "platform/GLCoreFunc.h"
- #include "platform/GLExtFunc.h"
- #undef GL_FUNCTION
- */
- #else
- // stub out these
- GLAPI void GLAPIENTRY glPointSize( GLfloat size )
- {
- }
- GLAPI void GLAPIENTRY glClipPlane( GLenum plane, const GLdouble *equation )
- {
- }
- GLAPI GLboolean GLAPIENTRY glAreTexturesResident( GLsizei n, const GLuint *textures, GLboolean *residences )
- {
- return true;
- }
- GLAPI void GLAPIENTRY glColor3i( GLint red, GLint green, GLint blue )
- {
- glColor3f(255.0f/red, 255.0f/green, 255.0f/blue);
- }
- GLAPI void GLAPIENTRY glRecti( GLint x1, GLint y1, GLint x2, GLint y2 )
- {
- //glRectf(x1, y2, x2, y2);
- }
- #endif
- void gluOrtho2D( GLfloat left, GLfloat right, GLfloat bottom, GLfloat top )
- {
- glOrtho( left, right, bottom, top, -1.0f, 1.0f );
- int glError;
- glError = TEST_FOR_OPENGL_ERRORS
- }
- GLint gluProject( GLdouble objx, GLdouble objy, GLdouble objz, const F64 *model, const F64 * proj, const GLint * vp, F64 * winx, F64 * winy, F64 * winz )
- {
- Vector4F v = Vector4F( objx, objy, objz, 1.0f );
- MatrixF pmat = MatrixF( false );
- for (int i=0; i<16; i++) { ((F32*)pmat)[i] = (float)proj[i]; }
- MatrixF mmat = MatrixF( false );
- for (int i=0; i<16; i++) { ((F32*)mmat)[i] = (float)model[i]; }
-
- //Luma: Projection fix
- mmat.transpose();
- pmat.transpose();
- (pmat.mul(mmat)).mul(v);
-
- //Luma: Projection fix
- if(v.w == 0.0f)
- {
- return GL_FALSE;
- }
- F32 invW = 1.0f / v.w;
- v.x *= invW;
- v.y *= invW;
- v.z *= invW;
-
- *winx = (GLfloat)vp[0] + (GLfloat)vp[2] * (v.x + 1.0f) * 0.5f;
- *winy = (GLfloat)vp[1] + (GLfloat)vp[3] * (v.y + 1.0f) * 0.5f;
- *winz = (v.z + 1.0f) * 0.5f;
- int glError;
- glError = TEST_FOR_OPENGL_ERRORS
- return GL_TRUE;
- }
- GLint gluUnProject( GLdouble winx, GLdouble winy, GLdouble winz, const F64 *model, const F64 * proj, const GLint * vp, F64 * x, F64 * y, F64 * z )
- {
- Vector4F v = Vector4F( 2.0f*(winx-vp[0])/vp[2] - 1.0f, 2.0f*(winy-vp[1])/vp[2] - 1.0f, 2.0f*vp[2] - 1.0f, 1.0f );
- MatrixF pmat = MatrixF( false );
- for (int i=0; i<16; i++) { ((F32*)pmat)[i] = (float)proj[i]; }
- MatrixF mmat = MatrixF( false );
- for (int i=0; i<16; i++) { ((F32*)mmat)[i] = (float)model[i]; }
- mmat = pmat.mul(mmat);
- mmat = mmat.inverse();
- mmat.mul( v );
- *x = v.x;
- *y = v.y;
- *z = v.z;
- int glError;
- glError = TEST_FOR_OPENGL_ERRORS
- return GL_TRUE;
- }
|