2010-06-14 23:56:29 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
2010-04-26 19:09:44 -07:00
|
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is mozilla.org code.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Mozilla Corporation.
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 2009
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Vladimir Vukicevic <vladimir@pobox.com>
|
|
|
|
* Mark Steele <mwsteele@gmail.com>
|
|
|
|
* Bas Schouten <bschouten@mozilla.com>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this file under
|
|
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
|
|
*
|
|
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include "prlink.h"
|
|
|
|
|
2010-07-01 09:30:38 -07:00
|
|
|
#include "gfxImageSurface.h"
|
2010-04-26 19:09:44 -07:00
|
|
|
#include "GLContext.h"
|
2010-05-17 21:04:22 -07:00
|
|
|
#include "GLContextProvider.h"
|
2010-04-26 19:09:44 -07:00
|
|
|
|
|
|
|
namespace mozilla {
|
|
|
|
namespace gl {
|
|
|
|
|
2010-05-17 21:04:22 -07:00
|
|
|
// define this here since it's global to GLContextProvider, not any
|
|
|
|
// specific implementation
|
|
|
|
const ContextFormat ContextFormat::BasicRGBA32Format(ContextFormat::BasicRGBA32);
|
|
|
|
|
2010-04-26 19:09:44 -07:00
|
|
|
#define MAX_SYMBOL_LENGTH 128
|
|
|
|
#define MAX_SYMBOL_NAMES 5
|
|
|
|
|
2010-05-17 21:04:22 -07:00
|
|
|
PRBool
|
2010-04-26 19:09:44 -07:00
|
|
|
LibrarySymbolLoader::OpenLibrary(const char *library)
|
|
|
|
{
|
|
|
|
PRLibSpec lspec;
|
|
|
|
lspec.type = PR_LibSpec_Pathname;
|
|
|
|
lspec.value.pathname = library;
|
|
|
|
|
|
|
|
mLibrary = PR_LoadLibraryWithFlags(lspec, PR_LD_LAZY | PR_LD_LOCAL);
|
|
|
|
if (!mLibrary)
|
2010-05-17 21:04:22 -07:00
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
2010-04-26 19:09:44 -07:00
|
|
|
|
2010-05-17 21:04:22 -07:00
|
|
|
PRBool
|
|
|
|
LibrarySymbolLoader::LoadSymbols(SymLoadStruct *firstStruct, PRBool tryplatform, const char *prefix)
|
|
|
|
{
|
|
|
|
return LoadSymbols(mLibrary, firstStruct, tryplatform ? mLookupFunc : nsnull, prefix);
|
2010-04-26 19:09:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
PRFuncPtr
|
2010-05-17 21:04:22 -07:00
|
|
|
LibrarySymbolLoader::LookupSymbol(PRLibrary *lib,
|
2010-06-14 23:56:29 -07:00
|
|
|
const char *sym,
|
|
|
|
PlatformLookupFunction lookupFunction)
|
2010-04-26 19:09:44 -07:00
|
|
|
{
|
|
|
|
PRFuncPtr res = 0;
|
|
|
|
|
|
|
|
// try finding it in the library directly, if we have one
|
2010-05-17 21:04:22 -07:00
|
|
|
if (lib) {
|
|
|
|
res = PR_FindFunctionSymbol(lib, sym);
|
2010-04-26 19:09:44 -07:00
|
|
|
}
|
|
|
|
|
2010-07-18 22:01:14 -07:00
|
|
|
// then try looking it up via the lookup symbol
|
|
|
|
if (!res && lookupFunction) {
|
|
|
|
res = lookupFunction(sym);
|
|
|
|
}
|
|
|
|
|
|
|
|
// finally just try finding it in the process
|
2010-04-26 19:09:44 -07:00
|
|
|
if (!res) {
|
|
|
|
PRLibrary *leakedLibRef;
|
|
|
|
res = PR_FindFunctionSymbolAndLibrary(sym, &leakedLibRef);
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
2010-05-17 21:04:22 -07:00
|
|
|
LibrarySymbolLoader::LoadSymbols(PRLibrary *lib,
|
2010-06-14 23:56:29 -07:00
|
|
|
SymLoadStruct *firstStruct,
|
|
|
|
PlatformLookupFunction lookupFunction,
|
|
|
|
const char *prefix)
|
2010-04-26 19:09:44 -07:00
|
|
|
{
|
|
|
|
char sbuf[MAX_SYMBOL_LENGTH * 2];
|
2010-07-18 22:01:14 -07:00
|
|
|
int failCount = 0;
|
2010-04-26 19:09:44 -07:00
|
|
|
|
|
|
|
SymLoadStruct *ss = firstStruct;
|
|
|
|
while (ss->symPointer) {
|
|
|
|
*ss->symPointer = 0;
|
|
|
|
|
|
|
|
for (int i = 0; i < MAX_SYMBOL_NAMES; i++) {
|
|
|
|
if (ss->symNames[i] == nsnull)
|
|
|
|
break;
|
|
|
|
|
|
|
|
const char *s = ss->symNames[i];
|
|
|
|
if (prefix && *prefix != 0) {
|
|
|
|
strcpy(sbuf, prefix);
|
|
|
|
strcat(sbuf, ss->symNames[i]);
|
|
|
|
s = sbuf;
|
|
|
|
}
|
|
|
|
|
2010-05-17 21:04:22 -07:00
|
|
|
PRFuncPtr p = LookupSymbol(lib, s, lookupFunction);
|
2010-04-26 19:09:44 -07:00
|
|
|
if (p) {
|
|
|
|
*ss->symPointer = p;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*ss->symPointer == 0) {
|
|
|
|
fprintf (stderr, "Can't find symbol '%s'\n", ss->symNames[0]);
|
2010-07-18 22:01:14 -07:00
|
|
|
failCount++;
|
2010-04-26 19:09:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
ss++;
|
|
|
|
}
|
|
|
|
|
2010-07-18 22:01:14 -07:00
|
|
|
return failCount == 0 ? PR_TRUE : PR_FALSE;
|
2010-04-26 19:09:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX - we should really know the ARB/EXT variants of these
|
|
|
|
* instead of only handling the symbol if it's exposed directly.
|
|
|
|
*/
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
GLContext::InitWithPrefix(const char *prefix, PRBool trygl)
|
|
|
|
{
|
|
|
|
if (mInitialized) {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
SymLoadStruct symbols[] = {
|
|
|
|
{ (PRFuncPtr*) &fActiveTexture, { "ActiveTexture", "ActiveTextureARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fAttachShader, { "AttachShader", "AttachShaderARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBindAttribLocation, { "BindAttribLocation", "BindAttribLocationARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBindBuffer, { "BindBuffer", "BindBufferARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBindTexture, { "BindTexture", "BindTextureARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBlendColor, { "BlendColor", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBlendEquation, { "BlendEquation", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBlendEquationSeparate, { "BlendEquationSeparate", "BlendEquationSeparateEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBlendFunc, { "BlendFunc", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBlendFuncSeparate, { "BlendFuncSeparate", "BlendFuncSeparateEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBufferData, { "BufferData", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBufferSubData, { "BufferSubData", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fClear, { "Clear", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fClearColor, { "ClearColor", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fClearStencil, { "ClearStencil", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fColorMask, { "ColorMask", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fCullFace, { "CullFace", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDetachShader, { "DetachShader", "DetachShaderARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDepthFunc, { "DepthFunc", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDepthMask, { "DepthMask", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDisable, { "Disable", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDisableVertexAttribArray, { "DisableVertexAttribArray", "DisableVertexAttribArrayARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDrawArrays, { "DrawArrays", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fDrawElements, { "DrawElements", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fEnable, { "Enable", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fEnableVertexAttribArray, { "EnableVertexAttribArray", "EnableVertexAttribArrayARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fFinish, { "Finish", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fFlush, { "Flush", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fFrontFace, { "FrontFace", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetActiveAttrib, { "GetActiveAttrib", "GetActiveAttribARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetActiveUniform, { "GetActiveUniform", "GetActiveUniformARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetAttachedShaders, { "GetAttachedShaders", "GetAttachedShadersARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetAttribLocation, { "GetAttribLocation", "GetAttribLocationARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetIntegerv, { "GetIntegerv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetFloatv, { "GetFloatv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetBooleanv, { "GetBooleanv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetBufferParameteriv, { "GetBufferParameteriv", "GetBufferParameterivARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetError, { "GetError", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetProgramiv, { "GetProgramiv", "GetProgramivARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetProgramInfoLog, { "GetProgramInfoLog", "GetProgramInfoLogARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fTexParameteri, { "TexParameteri", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fTexParameterf, { "TexParameterf", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetString, { "GetString", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetTexParameterfv, { "GetTexParameterfv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetTexParameteriv, { "GetTexParameteriv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetUniformfv, { "GetUniformfv", "GetUniformfvARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetUniformiv, { "GetUniformiv", "GetUniformivARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetUniformLocation, { "GetUniformLocation", "GetUniformLocationARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetVertexAttribfv, { "GetVertexAttribfv", "GetVertexAttribfvARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetVertexAttribiv, { "GetVertexAttribiv", "GetVertexAttribivARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fHint, { "Hint", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsBuffer, { "IsBuffer", "IsBufferARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsEnabled, { "IsEnabled", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsProgram, { "IsProgram", "IsProgramARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsShader, { "IsShader", "IsShaderARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsTexture, { "IsTexture", "IsTextureARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fLineWidth, { "LineWidth", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fLinkProgram, { "LinkProgram", "LinkProgramARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fPixelStorei, { "PixelStorei", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fPolygonOffset, { "PolygonOffset", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fReadPixels, { "ReadPixels", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fSampleCoverage, { "SampleCoverage", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fScissor, { "Scissor", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilFunc, { "StencilFunc", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilFuncSeparate, { "StencilFuncSeparate", "StencilFuncSeparateEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilMask, { "StencilMask", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilMaskSeparate, { "StencilMaskSeparate", "StencilMaskSeparateEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilOp, { "StencilOp", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fStencilOpSeparate, { "StencilOpSeparate", "StencilOpSeparateEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fTexImage2D, { "TexImage2D", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fTexSubImage2D, { "TexSubImage2D", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform1f, { "Uniform1f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform1fv, { "Uniform1fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform1i, { "Uniform1i", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform1iv, { "Uniform1iv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform2f, { "Uniform2f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform2fv, { "Uniform2fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform2i, { "Uniform2i", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform2iv, { "Uniform2iv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform3f, { "Uniform3f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform3fv, { "Uniform3fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform3i, { "Uniform3i", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform3iv, { "Uniform3iv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform4f, { "Uniform4f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform4fv, { "Uniform4fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform4i, { "Uniform4i", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniform4iv, { "Uniform4iv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniformMatrix2fv, { "UniformMatrix2fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniformMatrix3fv, { "UniformMatrix3fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUniformMatrix4fv, { "UniformMatrix4fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fUseProgram, { "UseProgram", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fValidateProgram, { "ValidateProgram", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttribPointer, { "VertexAttribPointer", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib1f, { "VertexAttrib1f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib2f, { "VertexAttrib2f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib3f, { "VertexAttrib3f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib4f, { "VertexAttrib4f", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib1fv, { "VertexAttrib1fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib2fv, { "VertexAttrib2fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib3fv, { "VertexAttrib3fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttrib4fv, { "VertexAttrib4fv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fViewport, { "Viewport", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fCompileShader, { "CompileShader", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fCopyTexImage2D, { "CopyTexImage2D", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fCopyTexSubImage2D, { "CopyTexSubImage2D", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetShaderiv, { "GetShaderiv", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetShaderInfoLog, { "GetShaderInfoLog", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetShaderSource, { "GetShaderSource", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fShaderSource, { "ShaderSource", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fVertexAttribPointer, { "VertexAttribPointer", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBindFramebuffer, { "BindFramebuffer", "BindFramebufferEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fBindRenderbuffer, { "BindRenderbuffer", "BindRenderbufferEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fCheckFramebufferStatus, { "CheckFramebufferStatus", "CheckFramebufferStatusEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fFramebufferRenderbuffer, { "FramebufferRenderbuffer", "FramebufferRenderbufferEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fFramebufferTexture2D, { "FramebufferTexture2D", "FramebufferTexture2DEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGenerateMipmap, { "GenerateMipmap", "GenerateMipmapEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetFramebufferAttachmentParameteriv, { "GetFramebufferAttachmentParameteriv", "GetFramebufferAttachmentParameterivEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fGetRenderbufferParameteriv, { "GetRenderbufferParameteriv", "GetRenderbufferParameterivEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsFramebuffer, { "IsFramebuffer", "IsFramebufferEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fIsRenderbuffer, { "IsRenderbuffer", "IsRenderbufferEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &fRenderbufferStorage, { "RenderbufferStorage", "RenderbufferStorageEXT", NULL } },
|
2010-07-18 22:01:14 -07:00
|
|
|
|
|
|
|
{ (PRFuncPtr*) &priv_fGenBuffers, { "GenBuffers", "GenBuffersARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fGenTextures, { "GenTextures", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fCreateProgram, { "CreateProgram", "CreateProgramARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fCreateShader, { "CreateShader", "CreateShaderARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fGenFramebuffers, { "GenFramebuffers", "GenFramebuffersEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fGenRenderbuffers, { "GenRenderbuffers", "GenRenderbuffersEXT", NULL } },
|
|
|
|
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteBuffers, { "DeleteBuffers", "DeleteBuffersARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteTextures, { "DeleteTextures", "DeleteTexturesARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteProgram, { "DeleteProgram", "DeleteProgramARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteShader, { "DeleteShader", "DeleteShaderARB", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteFramebuffers, { "DeleteFramebuffers", "DeleteFramebuffersEXT", NULL } },
|
|
|
|
{ (PRFuncPtr*) &priv_fDeleteRenderbuffers, { "DeleteRenderbuffers", "DeleteRenderbuffersEXT", NULL } },
|
2010-04-26 19:09:44 -07:00
|
|
|
|
2010-07-28 14:24:09 -07:00
|
|
|
{ mIsGLES2 ? (PRFuncPtr*) &priv_fClearDepthf : (PRFuncPtr*) &priv_fClearDepth,
|
|
|
|
{ mIsGLES2 ? "ClearDepthf" : "ClearDepth", NULL } },
|
|
|
|
{ mIsGLES2 ? (PRFuncPtr*) &priv_fDepthRangef : (PRFuncPtr*) &priv_fDepthRange,
|
|
|
|
{ mIsGLES2 ? "DepthRangef" : "DepthRange", NULL } },
|
|
|
|
|
|
|
|
// XXX FIXME -- we shouldn't be using glReadBuffer!
|
|
|
|
{ mIsGLES2 ? (PRFuncPtr*) NULL : (PRFuncPtr*) &fReadBuffer,
|
|
|
|
{ mIsGLES2 ? NULL : "ReadBuffer", NULL } },
|
|
|
|
|
2010-04-26 19:09:44 -07:00
|
|
|
{ NULL, { NULL } },
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
mInitialized = LoadSymbols(&symbols[0], trygl, prefix);
|
2010-08-06 22:09:18 -07:00
|
|
|
|
|
|
|
if (mInitialized) {
|
|
|
|
InitExtensions();
|
|
|
|
}
|
|
|
|
|
2010-04-26 19:09:44 -07:00
|
|
|
return mInitialized;
|
|
|
|
}
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
// should match the order of GLExtensions
|
|
|
|
static const char *sExtensionNames[] = {
|
|
|
|
"GL_EXT_framebuffer_object",
|
|
|
|
"GL_ARB_framebuffer_object",
|
2010-08-06 22:09:18 -07:00
|
|
|
"GL_ARB_texture_rectangle",
|
2010-08-06 22:09:18 -07:00
|
|
|
"GL_EXT_bgra",
|
|
|
|
"GL_EXT_texture_format_BGRA8888",
|
|
|
|
"GL_OES_depth24",
|
|
|
|
"GL_OES_depth32",
|
|
|
|
"GL_OES_stencil8",
|
|
|
|
"GL_OES_texture_npot",
|
|
|
|
"GL_OES_depth_texture",
|
|
|
|
"GL_OES_packed_depth_stencil",
|
|
|
|
"GL_IMG_read_format",
|
|
|
|
"GL_EXT_read_format_bgra",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::InitExtensions()
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
const GLubyte *extensions = fGetString(LOCAL_GL_EXTENSIONS);
|
|
|
|
char *exts = strdup((char *)extensions);
|
|
|
|
|
|
|
|
printf_stderr("GL extensions: %s\n", exts);
|
|
|
|
|
|
|
|
char *s = exts;
|
|
|
|
bool done = false;
|
|
|
|
while (!done) {
|
|
|
|
char *space = strchr(s, ' ');
|
|
|
|
if (space) {
|
|
|
|
*space = '\0';
|
|
|
|
} else {
|
|
|
|
done = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; sExtensionNames[i]; ++i) {
|
|
|
|
if (strcmp(s, sExtensionNames[i]) == 0) {
|
|
|
|
printf_stderr("Found extension %s\n", s);
|
|
|
|
mAvailableExtensions[i] = 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
s = space+1;
|
|
|
|
}
|
|
|
|
|
|
|
|
free(exts);
|
|
|
|
}
|
|
|
|
|
2010-06-14 23:55:08 -07:00
|
|
|
PRBool
|
|
|
|
GLContext::IsExtensionSupported(const char *extension)
|
|
|
|
{
|
|
|
|
const GLubyte *extensions = NULL;
|
|
|
|
const GLubyte *start;
|
|
|
|
GLubyte *where, *terminator;
|
|
|
|
|
|
|
|
/* Extension names should not have spaces. */
|
|
|
|
where = (GLubyte *) strchr(extension, ' ');
|
|
|
|
if (where || *extension == '\0')
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
extensions = fGetString(LOCAL_GL_EXTENSIONS);
|
|
|
|
/*
|
|
|
|
* It takes a bit of care to be fool-proof about parsing the
|
|
|
|
* OpenGL extensions string. Don't be fooled by sub-strings,
|
|
|
|
* etc.
|
|
|
|
*/
|
|
|
|
start = extensions;
|
|
|
|
for (;;) {
|
|
|
|
where = (GLubyte *) strstr((const char *) start, extension);
|
|
|
|
if (!where) {
|
2010-06-14 23:56:29 -07:00
|
|
|
break;
|
2010-06-14 23:55:08 -07:00
|
|
|
}
|
|
|
|
terminator = where + strlen(extension);
|
|
|
|
if (where == start || *(where - 1) == ' ') {
|
2010-06-14 23:56:29 -07:00
|
|
|
if (*terminator == ' ' || *terminator == '\0') {
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
2010-06-14 23:55:08 -07:00
|
|
|
}
|
|
|
|
start = terminator;
|
|
|
|
}
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-07-01 09:30:38 -07:00
|
|
|
already_AddRefed<TextureImage>
|
|
|
|
GLContext::CreateTextureImage(const nsIntSize& aSize,
|
|
|
|
TextureImage::ContentType aContentType,
|
|
|
|
GLint aWrapMode,
|
|
|
|
PRBool aUseNearestFilter)
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
|
|
|
|
GLuint texture;
|
|
|
|
fGenTextures(1, &texture);
|
|
|
|
|
|
|
|
fActiveTexture(LOCAL_GL_TEXTURE0);
|
|
|
|
fBindTexture(LOCAL_GL_TEXTURE_2D, texture);
|
|
|
|
|
|
|
|
GLint texfilter = aUseNearestFilter ? LOCAL_GL_NEAREST : LOCAL_GL_LINEAR;
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MIN_FILTER, texfilter);
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MAG_FILTER, texfilter);
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_WRAP_S, aWrapMode);
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_WRAP_T, aWrapMode);
|
|
|
|
DEBUG_GL_ERROR_CHECK(this);
|
|
|
|
|
|
|
|
return CreateBasicTextureImage(texture, aSize, aContentType, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
BasicTextureImage::~BasicTextureImage()
|
|
|
|
{
|
2010-07-19 21:05:42 -07:00
|
|
|
nsRefPtr<GLContext> ctx = mGLContext->GetSharedContext();
|
|
|
|
if (!ctx) {
|
|
|
|
ctx = mGLContext;
|
|
|
|
}
|
|
|
|
ctx->MakeCurrent();
|
|
|
|
ctx->fDeleteTextures(1, &mTexture);
|
2010-07-01 09:30:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
gfxContext*
|
|
|
|
BasicTextureImage::BeginUpdate(nsIntRegion& aRegion)
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!mUpdateContext, "BeginUpdate() without EndUpdate()?");
|
|
|
|
|
|
|
|
// determine the region the client will need to repaint
|
|
|
|
if (!mTextureInited)
|
|
|
|
// if the texture hasn't been initialized yet, force the
|
|
|
|
// client to paint everything
|
|
|
|
mUpdateRect = nsIntRect(nsIntPoint(0, 0), mSize);
|
|
|
|
else
|
|
|
|
mUpdateRect = aRegion.GetBounds();
|
|
|
|
// the basic impl can't upload updates to disparate regions,
|
|
|
|
// only rects
|
|
|
|
aRegion = nsIntRegion(mUpdateRect);
|
|
|
|
|
|
|
|
nsIntSize rgnSize = mUpdateRect.Size();
|
2010-07-19 14:54:17 -07:00
|
|
|
if (!nsIntRect(nsIntPoint(0, 0), mSize).Contains(mUpdateRect)) {
|
2010-07-01 09:30:38 -07:00
|
|
|
NS_ERROR("update outside of image");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
ImageFormat format =
|
|
|
|
(GetContentType() == gfxASurface::CONTENT_COLOR) ?
|
|
|
|
gfxASurface::ImageFormatRGB24 : gfxASurface::ImageFormatARGB32;
|
|
|
|
nsRefPtr<gfxASurface> updateSurface =
|
|
|
|
CreateUpdateSurface(gfxIntSize(rgnSize.width, rgnSize.height),
|
|
|
|
format);
|
|
|
|
if (!updateSurface)
|
|
|
|
return NULL;
|
|
|
|
|
2010-07-19 14:54:17 -07:00
|
|
|
updateSurface->SetDeviceOffset(gfxPoint(-mUpdateRect.x, -mUpdateRect.y));
|
|
|
|
|
2010-07-01 09:30:38 -07:00
|
|
|
mUpdateContext = new gfxContext(updateSurface);
|
|
|
|
return mUpdateContext;
|
|
|
|
}
|
|
|
|
|
|
|
|
PRBool
|
|
|
|
BasicTextureImage::EndUpdate()
|
|
|
|
{
|
|
|
|
NS_ASSERTION(!!mUpdateContext, "EndUpdate() without BeginUpdate()?");
|
|
|
|
|
|
|
|
// FIXME: this is the slow boat. Make me fast (with GLXPixmap?).
|
2010-07-19 14:54:17 -07:00
|
|
|
nsRefPtr<gfxASurface> originalSurface = mUpdateContext->OriginalSurface();
|
|
|
|
|
|
|
|
// Undo the device offset that BeginUpdate set; doesn't much matter for us here,
|
|
|
|
// but important if we ever do anything directly with the surface.
|
|
|
|
originalSurface->SetDeviceOffset(gfxPoint(0, 0));
|
|
|
|
|
|
|
|
nsRefPtr<gfxImageSurface> uploadImage = GetImageForUpload(originalSurface);
|
2010-07-01 09:30:38 -07:00
|
|
|
if (!uploadImage)
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
mGLContext->fBindTexture(LOCAL_GL_TEXTURE_2D, mTexture);
|
|
|
|
if (!mTextureInited)
|
|
|
|
{
|
|
|
|
mGLContext->fTexImage2D(LOCAL_GL_TEXTURE_2D,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGBA,
|
|
|
|
mUpdateRect.width,
|
|
|
|
mUpdateRect.height,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGBA,
|
|
|
|
LOCAL_GL_UNSIGNED_BYTE,
|
|
|
|
uploadImage->Data());
|
|
|
|
mTextureInited = PR_TRUE;
|
|
|
|
} else {
|
|
|
|
mGLContext->fTexSubImage2D(LOCAL_GL_TEXTURE_2D,
|
|
|
|
0,
|
|
|
|
mUpdateRect.x,
|
|
|
|
mUpdateRect.y,
|
|
|
|
mUpdateRect.width,
|
|
|
|
mUpdateRect.height,
|
|
|
|
LOCAL_GL_RGBA,
|
|
|
|
LOCAL_GL_UNSIGNED_BYTE,
|
|
|
|
uploadImage->Data());
|
|
|
|
}
|
|
|
|
mUpdateContext = NULL;
|
|
|
|
return PR_TRUE; // mTexture is bound
|
|
|
|
}
|
|
|
|
|
2010-07-18 22:01:14 -07:00
|
|
|
PRBool
|
|
|
|
GLContext::ResizeOffscreenFBO(const gfxIntSize& aSize)
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
|
|
|
|
bool alpha = mCreationFormat.alpha > 0;
|
|
|
|
int depth = mCreationFormat.depth;
|
|
|
|
int stencil = mCreationFormat.stencil;
|
|
|
|
|
|
|
|
bool firstTime = (mOffscreenFBO == 0);
|
|
|
|
|
|
|
|
GLuint curBoundTexture = 0;
|
|
|
|
GLuint curBoundRenderbuffer = 0;
|
|
|
|
GLuint curBoundFramebuffer = 0;
|
|
|
|
|
|
|
|
GLint viewport[4];
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
bool useDepthStencil =
|
|
|
|
!mIsGLES2 || IsExtensionSupported(OES_packed_depth_stencil);
|
|
|
|
|
2010-07-18 22:01:14 -07:00
|
|
|
// save a few things for later restoring
|
|
|
|
fGetIntegerv(LOCAL_GL_TEXTURE_BINDING_2D, (GLint*) &curBoundTexture);
|
|
|
|
fGetIntegerv(LOCAL_GL_FRAMEBUFFER_BINDING, (GLint*) &curBoundFramebuffer);
|
|
|
|
fGetIntegerv(LOCAL_GL_RENDERBUFFER_BINDING, (GLint*) &curBoundRenderbuffer);
|
|
|
|
fGetIntegerv(LOCAL_GL_VIEWPORT, viewport);
|
|
|
|
|
|
|
|
// If this is the first time we're going through this, we need
|
|
|
|
// to create the objects we'll use. Otherwise, just bind them.
|
|
|
|
if (firstTime) {
|
|
|
|
fGenTextures(1, &mOffscreenTexture);
|
|
|
|
fBindTexture(LOCAL_GL_TEXTURE_2D, mOffscreenTexture);
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MIN_FILTER, LOCAL_GL_LINEAR);
|
|
|
|
fTexParameteri(LOCAL_GL_TEXTURE_2D, LOCAL_GL_TEXTURE_MAG_FILTER, LOCAL_GL_LINEAR);
|
|
|
|
|
|
|
|
fGenFramebuffers(1, &mOffscreenFBO);
|
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, mOffscreenFBO);
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
if (depth && stencil && useDepthStencil) {
|
2010-07-18 22:01:14 -07:00
|
|
|
fGenRenderbuffers(1, &mOffscreenDepthRB);
|
2010-07-28 14:24:09 -07:00
|
|
|
} else {
|
|
|
|
if (depth) {
|
|
|
|
fGenRenderbuffers(1, &mOffscreenDepthRB);
|
|
|
|
}
|
2010-07-18 22:01:14 -07:00
|
|
|
|
2010-07-28 14:24:09 -07:00
|
|
|
if (stencil) {
|
|
|
|
fGenRenderbuffers(1, &mOffscreenStencilRB);
|
|
|
|
}
|
2010-07-18 22:01:14 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fBindTexture(LOCAL_GL_TEXTURE_2D, mOffscreenTexture);
|
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, mOffscreenFBO);
|
|
|
|
}
|
|
|
|
|
|
|
|
// resize the FBO components
|
|
|
|
if (alpha) {
|
|
|
|
fTexImage2D(LOCAL_GL_TEXTURE_2D,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGBA,
|
|
|
|
aSize.width, aSize.height,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGBA,
|
|
|
|
LOCAL_GL_UNSIGNED_BYTE,
|
|
|
|
NULL);
|
|
|
|
} else {
|
|
|
|
fTexImage2D(LOCAL_GL_TEXTURE_2D,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGB,
|
|
|
|
aSize.width, aSize.height,
|
|
|
|
0,
|
|
|
|
LOCAL_GL_RGB,
|
2010-08-06 22:09:18 -07:00
|
|
|
#ifdef XP_WIN
|
|
|
|
LOCAL_GL_UNSIGNED_BYTE,
|
|
|
|
#else
|
2010-07-28 14:24:09 -07:00
|
|
|
mIsGLES2 ? LOCAL_GL_UNSIGNED_SHORT_5_6_5
|
2010-07-18 22:01:14 -07:00
|
|
|
: LOCAL_GL_UNSIGNED_BYTE,
|
2010-08-06 22:09:18 -07:00
|
|
|
#endif
|
2010-07-18 22:01:14 -07:00
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
if (depth && stencil && useDepthStencil) {
|
2010-07-18 22:01:14 -07:00
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, mOffscreenDepthRB);
|
|
|
|
fRenderbufferStorage(LOCAL_GL_RENDERBUFFER,
|
2010-07-28 14:24:09 -07:00
|
|
|
LOCAL_GL_DEPTH24_STENCIL8,
|
2010-07-18 22:01:14 -07:00
|
|
|
aSize.width, aSize.height);
|
2010-07-28 14:24:09 -07:00
|
|
|
} else {
|
|
|
|
if (depth) {
|
2010-08-06 22:09:18 -07:00
|
|
|
GLenum depthType;
|
|
|
|
if (mIsGLES2) {
|
|
|
|
if (IsExtensionSupported(OES_depth32)) {
|
|
|
|
depthType = LOCAL_GL_DEPTH_COMPONENT32;
|
|
|
|
} else if (IsExtensionSupported(OES_depth24)) {
|
|
|
|
depthType = LOCAL_GL_DEPTH_COMPONENT24;
|
|
|
|
} else {
|
|
|
|
depthType = LOCAL_GL_DEPTH_COMPONENT16;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
depthType = LOCAL_GL_DEPTH_COMPONENT24;
|
|
|
|
}
|
|
|
|
|
2010-07-28 14:24:09 -07:00
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, mOffscreenDepthRB);
|
|
|
|
fRenderbufferStorage(LOCAL_GL_RENDERBUFFER,
|
|
|
|
mIsGLES2 ? LOCAL_GL_DEPTH_COMPONENT16
|
|
|
|
: LOCAL_GL_DEPTH_COMPONENT24,
|
|
|
|
aSize.width, aSize.height);
|
|
|
|
}
|
2010-07-18 22:01:14 -07:00
|
|
|
|
2010-07-28 14:24:09 -07:00
|
|
|
if (stencil) {
|
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, mOffscreenStencilRB);
|
|
|
|
fRenderbufferStorage(LOCAL_GL_RENDERBUFFER,
|
|
|
|
LOCAL_GL_STENCIL_INDEX8,
|
|
|
|
aSize.width, aSize.height);
|
|
|
|
}
|
2010-07-18 22:01:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// Now assemble the FBO, if we're creating one
|
|
|
|
// for the first time.
|
|
|
|
if (firstTime) {
|
|
|
|
fFramebufferTexture2D(LOCAL_GL_FRAMEBUFFER,
|
|
|
|
LOCAL_GL_COLOR_ATTACHMENT0,
|
|
|
|
LOCAL_GL_TEXTURE_2D,
|
|
|
|
mOffscreenTexture,
|
|
|
|
0);
|
2010-07-28 14:24:09 -07:00
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
if (depth && stencil && useDepthStencil) {
|
2010-07-18 22:01:14 -07:00
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER,
|
|
|
|
LOCAL_GL_DEPTH_ATTACHMENT,
|
|
|
|
LOCAL_GL_RENDERBUFFER,
|
|
|
|
mOffscreenDepthRB);
|
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER,
|
|
|
|
LOCAL_GL_STENCIL_ATTACHMENT,
|
|
|
|
LOCAL_GL_RENDERBUFFER,
|
2010-07-28 14:24:09 -07:00
|
|
|
mOffscreenDepthRB);
|
|
|
|
} else {
|
|
|
|
if (depth) {
|
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER,
|
|
|
|
LOCAL_GL_DEPTH_ATTACHMENT,
|
|
|
|
LOCAL_GL_RENDERBUFFER,
|
|
|
|
mOffscreenDepthRB);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stencil) {
|
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER,
|
|
|
|
LOCAL_GL_STENCIL_ATTACHMENT,
|
|
|
|
LOCAL_GL_RENDERBUFFER,
|
|
|
|
mOffscreenStencilRB);
|
|
|
|
}
|
2010-07-18 22:01:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We should be all resized. Check for framebuffer completeness.
|
|
|
|
GLenum status = fCheckFramebufferStatus(LOCAL_GL_FRAMEBUFFER);
|
|
|
|
if (status != LOCAL_GL_FRAMEBUFFER_COMPLETE) {
|
|
|
|
NS_WARNING("Error resizing offscreen framebuffer -- framebuffer not complete");
|
|
|
|
return PR_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
mOffscreenSize = aSize;
|
|
|
|
mOffscreenActualSize = aSize;
|
|
|
|
|
|
|
|
if (firstTime) {
|
|
|
|
UpdateActualFormat();
|
2010-08-06 22:09:18 -07:00
|
|
|
|
|
|
|
printf_stderr("Created offscreen FBO: r: %d g: %d b: %d a: %d depth: %d stencil: %d\n",
|
|
|
|
mActualFormat.red, mActualFormat.green, mActualFormat.blue, mActualFormat.alpha,
|
|
|
|
mActualFormat.depth, mActualFormat.stencil);
|
2010-07-18 22:01:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// We're good, and the framebuffer is already attached, so let's
|
|
|
|
// clear out our new framebuffer; otherwise we'll end up displaying
|
|
|
|
// random memory. We saved all of these things earlier so that we
|
|
|
|
// can restore them.
|
|
|
|
fViewport(0, 0, aSize.width, aSize.height);
|
|
|
|
|
|
|
|
// Ok, now restore the GL state back to what it was before the resize took place.
|
|
|
|
fBindTexture(LOCAL_GL_TEXTURE_2D, curBoundTexture);
|
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, curBoundRenderbuffer);
|
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, curBoundFramebuffer);
|
|
|
|
|
|
|
|
// -don't- restore the viewport the first time through this, since
|
|
|
|
// the previous one isn't valid.
|
|
|
|
if (!firstTime)
|
|
|
|
fViewport(viewport[0], viewport[1], viewport[2], viewport[3]);
|
|
|
|
|
|
|
|
ClearSafely();
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeleteOffscreenFBO()
|
|
|
|
{
|
|
|
|
fDeleteFramebuffers(1, &mOffscreenFBO);
|
|
|
|
fDeleteTextures(1, &mOffscreenTexture);
|
|
|
|
fDeleteRenderbuffers(1, &mOffscreenDepthRB);
|
|
|
|
fDeleteRenderbuffers(1, &mOffscreenStencilRB);
|
|
|
|
|
|
|
|
mOffscreenFBO = 0;
|
|
|
|
mOffscreenTexture = 0;
|
|
|
|
mOffscreenDepthRB = 0;
|
|
|
|
mOffscreenStencilRB = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::ClearSafely()
|
|
|
|
{
|
|
|
|
GLfloat clearColor[4];
|
|
|
|
GLfloat clearDepth;
|
|
|
|
GLint clearStencil;
|
|
|
|
|
|
|
|
fGetFloatv(LOCAL_GL_COLOR_CLEAR_VALUE, clearColor);
|
|
|
|
fGetFloatv(LOCAL_GL_DEPTH_CLEAR_VALUE, &clearDepth);
|
|
|
|
fGetIntegerv(LOCAL_GL_STENCIL_CLEAR_VALUE, &clearStencil);
|
|
|
|
|
|
|
|
fClearColor(0.0f, 0.0f, 0.0f, 0.0f);
|
|
|
|
fClearStencil(0);
|
2010-07-28 14:24:09 -07:00
|
|
|
fClearDepth(1.0f);
|
2010-07-18 22:01:14 -07:00
|
|
|
|
|
|
|
fClear(LOCAL_GL_COLOR_BUFFER_BIT | LOCAL_GL_DEPTH_BUFFER_BIT | LOCAL_GL_STENCIL_BUFFER_BIT);
|
|
|
|
|
|
|
|
fClearColor(clearColor[0], clearColor[1], clearColor[2], clearColor[3]);
|
|
|
|
fClearStencil(clearStencil);
|
|
|
|
fClearDepth(clearDepth);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::UpdateActualFormat()
|
|
|
|
{
|
2010-08-06 22:09:18 -07:00
|
|
|
ContextFormat nf;
|
|
|
|
|
|
|
|
fGetIntegerv(LOCAL_GL_RED_BITS, (GLint*) &nf.alpha);
|
|
|
|
fGetIntegerv(LOCAL_GL_GREEN_BITS, (GLint*) &nf.alpha);
|
|
|
|
fGetIntegerv(LOCAL_GL_BLUE_BITS, (GLint*) &nf.alpha);
|
|
|
|
fGetIntegerv(LOCAL_GL_ALPHA_BITS, (GLint*) &nf.alpha);
|
|
|
|
fGetIntegerv(LOCAL_GL_DEPTH_BITS, (GLint*) &nf.depth);
|
|
|
|
fGetIntegerv(LOCAL_GL_STENCIL_BITS, (GLint*) &nf.depth);
|
|
|
|
|
|
|
|
mActualFormat = nf;
|
2010-07-18 22:01:14 -07:00
|
|
|
}
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
void
|
|
|
|
GLContext::MarkDestroyed()
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
DeleteOffscreenFBO();
|
|
|
|
memset(&mFunctionListStartSentinel, 0, &mFunctionListEndSentinel - &mFunctionListStartSentinel);
|
|
|
|
}
|
|
|
|
|
|
|
|
already_AddRefed<gfxImageSurface>
|
|
|
|
GLContext::ReadTextureImage(GLuint aTexture,
|
|
|
|
const gfxIntSize& aSize,
|
|
|
|
GLenum aTextureFormat)
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
|
|
|
|
nsRefPtr<gfxImageSurface> isurf;
|
|
|
|
|
|
|
|
GLint oldrb, oldfb, oldprog, oldvp[4], oldPackAlignment;
|
|
|
|
GLint success;
|
|
|
|
|
|
|
|
GLuint rb = 0, fb = 0;
|
|
|
|
GLuint vs = 0, fs = 0, prog = 0;
|
|
|
|
|
|
|
|
const char *vShader =
|
|
|
|
"attribute vec4 aVertex;\n"
|
|
|
|
"attribute vec2 aTexCoord;\n"
|
|
|
|
"varying vec2 vTexCoord;\n"
|
|
|
|
"void main() { gl_Position = aVertex; vTexCoord = aTexCoord; }";
|
|
|
|
const char *fShader =
|
|
|
|
"#ifdef GL_ES\n"
|
|
|
|
"precision mediump float;\n"
|
|
|
|
"#endif\n"
|
|
|
|
"varying vec2 vTexCoord;\n"
|
|
|
|
"uniform sampler2D uTexture;\n"
|
|
|
|
"void main() { gl_FragColor = texture2D(uTexture, vTexCoord); }";
|
|
|
|
|
|
|
|
float verts[4*4] = {
|
|
|
|
-1.0f, -1.0f, 0.0f, 1.0f,
|
|
|
|
1.0f, -1.0f, 0.0f, 1.0f,
|
|
|
|
-1.0f, 1.0f, 0.0f, 1.0f,
|
|
|
|
1.0f, 1.0f, 0.0f, 1.0f
|
|
|
|
};
|
|
|
|
|
|
|
|
float texcoords[2*4] = {
|
|
|
|
0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f
|
|
|
|
};
|
|
|
|
|
|
|
|
fGetIntegerv(LOCAL_GL_RENDERBUFFER_BINDING, &oldrb);
|
|
|
|
fGetIntegerv(LOCAL_GL_FRAMEBUFFER_BINDING, &oldfb);
|
|
|
|
fGetIntegerv(LOCAL_GL_CURRENT_PROGRAM, &oldprog);
|
|
|
|
fGetIntegerv(LOCAL_GL_VIEWPORT, oldvp);
|
|
|
|
fGetIntegerv(LOCAL_GL_PACK_ALIGNMENT, &oldPackAlignment);
|
|
|
|
|
|
|
|
fGenRenderbuffers(1, &rb);
|
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, rb);
|
|
|
|
fRenderbufferStorage(LOCAL_GL_RENDERBUFFER, LOCAL_GL_RGBA,
|
|
|
|
aSize.width, aSize.height);
|
|
|
|
|
|
|
|
fGenFramebuffers(1, &fb);
|
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, fb);
|
|
|
|
fFramebufferRenderbuffer(LOCAL_GL_FRAMEBUFFER, LOCAL_GL_COLOR_ATTACHMENT0,
|
|
|
|
LOCAL_GL_RENDERBUFFER, rb);
|
|
|
|
|
|
|
|
if (fCheckFramebufferStatus(LOCAL_GL_FRAMEBUFFER) !=
|
|
|
|
LOCAL_GL_FRAMEBUFFER_COMPLETE)
|
|
|
|
{
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
vs = fCreateShader(LOCAL_GL_VERTEX_SHADER);
|
|
|
|
fs = fCreateShader(LOCAL_GL_FRAGMENT_SHADER);
|
|
|
|
fShaderSource(vs, 1, (const GLchar**) &vShader, NULL);
|
|
|
|
fShaderSource(fs, 1, (const GLchar**) &fShader, NULL);
|
|
|
|
prog = fCreateProgram();
|
|
|
|
fAttachShader(prog, vs);
|
|
|
|
fAttachShader(prog, fs);
|
|
|
|
fBindAttribLocation(prog, 0, "aVertex");
|
|
|
|
fBindAttribLocation(prog, 1, "aTexCoord");
|
|
|
|
fLinkProgram(prog);
|
|
|
|
|
|
|
|
fGetProgramiv(prog, LOCAL_GL_LINK_STATUS, &success);
|
|
|
|
if (!success) {
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
fUseProgram(prog);
|
|
|
|
|
|
|
|
fEnableVertexAttribArray(0);
|
|
|
|
fEnableVertexAttribArray(1);
|
|
|
|
|
|
|
|
fVertexAttribPointer(0, 4, LOCAL_GL_FLOAT, LOCAL_GL_FALSE, 0, verts);
|
|
|
|
fVertexAttribPointer(1, 2, LOCAL_GL_FLOAT, LOCAL_GL_FALSE, 0, texcoords);
|
|
|
|
|
|
|
|
fActiveTexture(LOCAL_GL_TEXTURE0);
|
|
|
|
fBindTexture(LOCAL_GL_TEXTURE_2D, aTexture);
|
|
|
|
|
|
|
|
fUniform1i(fGetUniformLocation(prog, "uTexture"), 0);
|
|
|
|
|
|
|
|
fViewport(0, 0, aSize.width, aSize.height);
|
|
|
|
|
|
|
|
fDrawArrays(LOCAL_GL_TRIANGLE_STRIP, 0, 4);
|
|
|
|
|
|
|
|
fDisableVertexAttribArray(1);
|
|
|
|
fDisableVertexAttribArray(0);
|
|
|
|
|
|
|
|
isurf = new gfxImageSurface(aSize, gfxASurface::ImageFormatARGB32);
|
|
|
|
if (!isurf || isurf->CairoStatus()) {
|
|
|
|
isurf = nsnull;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldPackAlignment != 4)
|
|
|
|
fPixelStorei(LOCAL_GL_PACK_ALIGNMENT, 4);
|
|
|
|
|
|
|
|
fReadPixels(0, 0, aSize.width, aSize.height,
|
|
|
|
LOCAL_GL_RGBA, LOCAL_GL_UNSIGNED_BYTE,
|
|
|
|
isurf->Data());
|
|
|
|
|
|
|
|
if (oldPackAlignment != 4)
|
|
|
|
fPixelStorei(LOCAL_GL_PACK_ALIGNMENT, oldPackAlignment);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
// note that deleting 0 has no effect in any of these calls
|
|
|
|
fDeleteRenderbuffers(1, &rb);
|
|
|
|
fDeleteFramebuffers(1, &fb);
|
|
|
|
fDeleteShader(vs);
|
|
|
|
fDeleteShader(fs);
|
|
|
|
fDeleteProgram(prog);
|
|
|
|
|
|
|
|
fBindRenderbuffer(LOCAL_GL_RENDERBUFFER, oldrb);
|
|
|
|
fBindFramebuffer(LOCAL_GL_FRAMEBUFFER, oldfb);
|
|
|
|
fUseProgram(oldprog);
|
|
|
|
fViewport(oldvp[0], oldvp[1], oldvp[2], oldvp[3]);
|
|
|
|
|
|
|
|
return isurf.forget();
|
|
|
|
}
|
|
|
|
|
2010-08-06 22:09:18 -07:00
|
|
|
void
|
|
|
|
GLContext::ReadPixelsIntoImageSurface(GLint aX, GLint aY,
|
|
|
|
GLsizei aWidth, GLsizei aHeight,
|
|
|
|
gfxImageSurface *aDest)
|
|
|
|
{
|
|
|
|
MakeCurrent();
|
|
|
|
|
|
|
|
if (aDest->Format() != gfxASurface::ImageFormatARGB32 &&
|
|
|
|
aDest->Format() != gfxASurface::ImageFormatRGB24)
|
|
|
|
{
|
|
|
|
NS_WARNING("ReadPixelsIntoImageSurface called with invalid image format");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aDest->Width() != aWidth ||
|
|
|
|
aDest->Height() != aHeight ||
|
|
|
|
aDest->Stride() != aWidth * 4)
|
|
|
|
{
|
|
|
|
NS_WARNING("ReadPixelsIntoImageSurface called with wrong size or stride surface");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLint currentPackAlignment = 0;
|
|
|
|
fGetIntegerv(LOCAL_GL_PACK_ALIGNMENT, ¤tPackAlignment);
|
|
|
|
fPixelStorei(LOCAL_GL_PACK_ALIGNMENT, 4);
|
|
|
|
|
|
|
|
// defaults for desktop
|
|
|
|
GLenum format = LOCAL_GL_BGRA;
|
|
|
|
GLenum datatype = LOCAL_GL_UNSIGNED_INT_8_8_8_8_REV;
|
|
|
|
bool swap = false;
|
|
|
|
|
|
|
|
if (IsGLES2()) {
|
|
|
|
datatype = LOCAL_GL_UNSIGNED_BYTE;
|
|
|
|
|
|
|
|
if (IsExtensionSupported(gl::GLContext::EXT_read_format_bgra) ||
|
|
|
|
IsExtensionSupported(gl::GLContext::IMG_read_format) ||
|
|
|
|
IsExtensionSupported(gl::GLContext::EXT_bgra))
|
|
|
|
{
|
|
|
|
format = LOCAL_GL_BGRA;
|
|
|
|
} else {
|
|
|
|
format = LOCAL_GL_RGBA;
|
|
|
|
swap = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fReadPixels(0, 0, aWidth, aHeight,
|
|
|
|
format, datatype,
|
|
|
|
aDest->Data());
|
|
|
|
|
|
|
|
if (swap) {
|
|
|
|
// swap B and R bytes
|
|
|
|
for (int j = 0; j < aHeight; ++j) {
|
|
|
|
PRUint32 *row = (PRUint32*) (aDest->Data() + aDest->Stride() * j);
|
|
|
|
for (int i = 0; i < aWidth; ++i) {
|
|
|
|
*row = (*row & 0xff00ff00) | ((*row & 0xff) << 16) | ((*row & 0xff0000) >> 16);
|
|
|
|
row++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fPixelStorei(LOCAL_GL_PACK_ALIGNMENT, currentPackAlignment);
|
|
|
|
}
|
|
|
|
|
2010-07-18 22:01:14 -07:00
|
|
|
#ifdef DEBUG
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedProgram(GLContext *aOrigin, GLuint aName)
|
|
|
|
{
|
|
|
|
mTrackedPrograms.AppendElement(NamedResource(aOrigin, aName));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedShader(GLContext *aOrigin, GLuint aName)
|
|
|
|
{
|
|
|
|
mTrackedShaders.AppendElement(NamedResource(aOrigin, aName));
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedBuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
for (GLsizei i = 0; i < aCount; ++i) {
|
|
|
|
mTrackedBuffers.AppendElement(NamedResource(aOrigin, aNames[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedTextures(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
for (GLsizei i = 0; i < aCount; ++i) {
|
|
|
|
mTrackedTextures.AppendElement(NamedResource(aOrigin, aNames[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedFramebuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
for (GLsizei i = 0; i < aCount; ++i) {
|
|
|
|
mTrackedFramebuffers.AppendElement(NamedResource(aOrigin, aNames[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::CreatedRenderbuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
for (GLsizei i = 0; i < aCount; ++i) {
|
|
|
|
mTrackedRenderbuffers.AppendElement(NamedResource(aOrigin, aNames[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
RemoveNamesFromArray(GLContext *aOrigin, GLsizei aCount, GLuint *aNames, nsTArray<GLContext::NamedResource>& aArray)
|
|
|
|
{
|
|
|
|
for (GLsizei j = 0; j < aCount; ++j) {
|
|
|
|
GLuint name = aNames[j];
|
|
|
|
PRBool found = PR_FALSE;
|
|
|
|
for (PRUint32 i = 0; i < aArray.Length(); ++i) {
|
|
|
|
if (aArray[i].name == name) {
|
|
|
|
aArray.RemoveElementAt(i);
|
|
|
|
found = PR_TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found)
|
|
|
|
printf_stderr("GL Context %p deleting resource %d, which doesn't exist!", aOrigin, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedProgram(GLContext *aOrigin, GLuint aName)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, 1, &aName, mTrackedPrograms);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedShader(GLContext *aOrigin, GLuint aName)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, 1, &aName, mTrackedShaders);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedBuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, aCount, aNames, mTrackedBuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedTextures(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, aCount, aNames, mTrackedTextures);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedFramebuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, aCount, aNames, mTrackedFramebuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::DeletedRenderbuffers(GLContext *aOrigin, GLsizei aCount, GLuint *aNames)
|
|
|
|
{
|
|
|
|
RemoveNamesFromArray(aOrigin, aCount, aNames, mTrackedRenderbuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
MarkContextDestroyedInArray(GLContext *aContext, nsTArray<GLContext::NamedResource>& aArray)
|
|
|
|
{
|
|
|
|
for (PRUint32 i = 0; i < aArray.Length(); ++i) {
|
|
|
|
if (aArray[i].origin == aContext)
|
|
|
|
aArray[i].originDeleted = PR_TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::SharedContextDestroyed(GLContext *aChild)
|
|
|
|
{
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedPrograms);
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedShaders);
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedTextures);
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedFramebuffers);
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedRenderbuffers);
|
|
|
|
MarkContextDestroyedInArray(aChild, mTrackedBuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ReportArrayContents(const nsTArray<GLContext::NamedResource>& aArray)
|
|
|
|
{
|
|
|
|
nsTArray<GLContext::NamedResource> copy(aArray);
|
|
|
|
copy.Sort();
|
|
|
|
|
|
|
|
GLContext *lastContext = NULL;
|
|
|
|
for (PRUint32 i = 0; i < copy.Length(); ++i) {
|
|
|
|
if (lastContext != copy[i].origin) {
|
|
|
|
if (lastContext)
|
|
|
|
printf_stderr("\n");
|
|
|
|
printf_stderr(" [%p - %s] ", copy[i].origin, copy[i].originDeleted ? "deleted" : "live");
|
|
|
|
lastContext = copy[i].origin;
|
|
|
|
}
|
|
|
|
printf_stderr("%d ", copy[i].name);
|
|
|
|
}
|
|
|
|
printf_stderr("\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
GLContext::ReportOutstandingNames()
|
|
|
|
{
|
|
|
|
printf_stderr("== GLContext %p ==\n", this);
|
|
|
|
printf_stderr("Outstanding Textures:\n");
|
|
|
|
ReportArrayContents(mTrackedTextures);
|
|
|
|
printf_stderr("Outstanding Buffers:\n");
|
|
|
|
ReportArrayContents(mTrackedBuffers);
|
|
|
|
printf_stderr("Outstanding Programs:\n");
|
|
|
|
ReportArrayContents(mTrackedPrograms);
|
|
|
|
printf_stderr("Outstanding Shaders:\n");
|
|
|
|
ReportArrayContents(mTrackedShaders);
|
|
|
|
printf_stderr("Outstanding Framebuffers:\n");
|
|
|
|
ReportArrayContents(mTrackedFramebuffers);
|
|
|
|
printf_stderr("Outstanding Renderbuffers:\n");
|
|
|
|
ReportArrayContents(mTrackedRenderbuffers);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* DEBUG */
|
|
|
|
|
2010-04-26 19:09:44 -07:00
|
|
|
} /* namespace gl */
|
|
|
|
} /* namespace mozilla */
|