b=565417; add OSMesa GLContextProvider (pbuffers only); r=vlad

This commit is contained in:
Benoit Jacob 2010-05-19 13:46:08 -07:00
parent 2b3ba55c20
commit 7dfee0fad7
5 changed files with 313 additions and 1 deletions

View File

@ -137,6 +137,15 @@ WebGLContext::SetDimensions(PRInt32 width, PRInt32 height)
gl = gl::sGLContextProvider.CreatePBuffer(gfxIntSize(width, height), format); gl = gl::sGLContextProvider.CreatePBuffer(gfxIntSize(width, height), format);
if (!gl) {
LogMessage("Canvas 3D: can't get a native PBuffer, trying OSMesa...");
gl = gl::GLContextProviderOSMesa::CreatePBuffer(gfxIntSize(width, height), format);
if (!gl) {
LogMessage("Canvas 3D: can't create a OSMesa pseudo-PBuffer.");
return NS_ERROR_FAILURE;
}
}
if (!ValidateGL()) { if (!ValidateGL()) {
LogMessage("Canvas 3D: Couldn't validate OpenGL implementation; is everything needed present?"); LogMessage("Canvas 3D: Couldn't validate OpenGL implementation; is everything needed present?");
return NS_ERROR_FAILURE; return NS_ERROR_FAILURE;
@ -285,7 +294,24 @@ WebGLContext::GetCanvasLayer(LayerManager *manager)
CanvasLayer::Data data; CanvasLayer::Data data;
data.mGLContext = gl.get(); // the gl context may either provide a native PBuffer, in which case we want to initialize
// data with the gl context directly, or may provide a surface to which it renders (this is the case
// of OSMesa contexts), in which case we want to initialize data with that surface.
void* native_pbuffer = gl->GetNativeData(gl::GLContext::NativePBuffer);
void* native_surface = gl->GetNativeData(gl::GLContext::NativeImageSurface);
if (native_pbuffer) {
data.mGLContext = gl.get();
}
else if (native_surface) {
data.mSurface = static_cast<gfxASurface*>(native_surface);
}
else {
NS_WARNING("The GLContext has neither a native PBuffer nor a native surface!");
return nsnull;
}
data.mSize = nsIntSize(mWidth, mHeight); data.mSize = nsIntSize(mWidth, mHeight);
data.mGLBufferIsPremultiplied = PR_FALSE; data.mGLBufferIsPremultiplied = PR_FALSE;

View File

@ -121,6 +121,7 @@ public:
NativeGLContext, NativeGLContext,
NativeCGLContext, NativeCGLContext,
NativePBuffer, NativePBuffer,
NativeImageSurface,
NativeDataTypeMax NativeDataTypeMax
}; };

View File

@ -121,6 +121,33 @@ public:
already_AddRefed<GLContext> CreateForWindow(nsIWidget *aWidget); already_AddRefed<GLContext> CreateForWindow(nsIWidget *aWidget);
}; };
/** Same as GLContextProvider but for off-screen Mesa rendering */
class THEBES_API GLContextProviderOSMesa
{
public:
typedef GLContextProvider::ContextFormat ContextFormat;
/**
* Creates a PBuffer.
*
* @param aSize Size of the pbuffer to create
* @param aFormat A ContextFormat describing the desired context attributes. Defaults to a basic RGBA32 context.
*
* @return Context to use for this Pbuffer
*/
static already_AddRefed<GLContext> CreatePBuffer(const gfxIntSize &aSize,
const ContextFormat& aFormat = ContextFormat::BasicRGBA32Format);
/**
* Create a context that renders to the surface of the widget that is
* passed in.
*
* @param Widget whose surface to create a context for
* @return Context to use for this window
*/
static already_AddRefed<GLContext> CreateForWindow(nsIWidget *aWidget);
};
extern GLContextProvider THEBES_API sGLContextProvider; extern GLContextProvider THEBES_API sGLContextProvider;
} }

View File

@ -0,0 +1,257 @@
/* -*- Mode: C++; tab-width: 20; indent-tabs-mode: nil; c-basic-offset: 4 -*-
* ***** 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 Initial Developer of the Original Code is Mozilla Foundation.
* Portions created by the Initial Developer are Copyright (C) 2010
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Benoit Jacob <bjacob@mozilla.com>
* Bas Schouten <bschouten@mozilla.com>
* Vladimir Vukicevic <vladimir@pobox.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 "GLContextProvider.h"
#include "GLContext.h"
#include "nsDebug.h"
#include "nsString.h"
#include "nsIWidget.h"
#include "nsDirectoryServiceUtils.h"
#include "nsAppDirectoryServiceDefs.h"
#include "nsIPrefService.h"
#include "gfxASurface.h"
#include "gfxImageSurface.h"
// from GL/osmesa.h. We don't include that file so as to avoid having a build-time dependency on OSMesa.
#define OSMESA_RGBA GL_RGBA
#define OSMESA_BGRA 0x1
#define OSMESA_ARGB 0x2
#define OSMESA_RGB GL_RGB
#define OSMESA_BGR 0x4
#define OSMESA_RGB_565 0x5
#define OSMESA_Y_UP 0x11
namespace mozilla {
namespace gl {
typedef void* PrivateOSMesaContext;
class OSMesaLibrary
{
public:
OSMesaLibrary() : mInitialized(PR_FALSE), mOSMesaLibrary(nsnull) {}
typedef PrivateOSMesaContext (GLAPIENTRY * PFNOSMESACREATECONTEXTEXT) (GLenum, GLint, GLint, GLint, PrivateOSMesaContext);
typedef void (GLAPIENTRY * PFNOSMESADESTROYCONTEXT) (PrivateOSMesaContext);
typedef bool (GLAPIENTRY * PFNOSMESAMAKECURRENT) (PrivateOSMesaContext, void *, GLenum, GLsizei, GLsizei);
typedef PrivateOSMesaContext (GLAPIENTRY * PFNOSMESAGETCURRENTCONTEXT) (void);
typedef void (GLAPIENTRY * PFNOSMESAPIXELSTORE) (GLint, GLint);
typedef PRFuncPtr (GLAPIENTRY * PFNOSMESAGETPROCADDRESS) (const char*);
PFNOSMESACREATECONTEXTEXT fCreateContextExt;
PFNOSMESADESTROYCONTEXT fDestroyContext;
PFNOSMESAMAKECURRENT fMakeCurrent;
PFNOSMESAGETCURRENTCONTEXT fGetCurrentContext;
PFNOSMESAPIXELSTORE fPixelStore;
PFNOSMESAGETPROCADDRESS fGetProcAddress;
PRBool EnsureInitialized();
private:
PRBool mInitialized;
PRLibrary *mOSMesaLibrary;
};
OSMesaLibrary sOSMesaLibrary;
PRBool
OSMesaLibrary::EnsureInitialized()
{
if (mInitialized)
return PR_TRUE;
nsresult rv;
nsCOMPtr<nsIPrefService> prefService = do_GetService(NS_PREFSERVICE_CONTRACTID, &rv);
NS_ENSURE_SUCCESS(rv, PR_FALSE);
nsCOMPtr<nsIPrefBranch> prefBranch;
rv = prefService->GetBranch("webgl.", getter_AddRefs(prefBranch));
NS_ENSURE_SUCCESS(rv, PR_FALSE);
nsCString osmesalib;
rv = prefBranch->GetCharPref("osmesalib", getter_Copies(osmesalib));
if (NS_FAILED(rv) ||
osmesalib.Length() == 0)
{
return PR_FALSE;
}
mOSMesaLibrary = PR_LoadLibrary(osmesalib.get());
if (!mOSMesaLibrary) {
NS_WARNING("Canvas 3D: Couldn't open OSMesa lib -- webgl.osmesalib path is incorrect, or not a valid shared library");
return PR_FALSE;
}
LibrarySymbolLoader::SymLoadStruct symbols[] = {
{ (PRFuncPtr*) &fCreateContextExt, { "OSMesaCreateContextExt", NULL } },
{ (PRFuncPtr*) &fMakeCurrent, { "OSMesaMakeCurrent", NULL } },
{ (PRFuncPtr*) &fPixelStore, { "OSMesaPixelStore", NULL } },
{ (PRFuncPtr*) &fDestroyContext, { "OSMesaDestroyContext", NULL } },
{ (PRFuncPtr*) &fGetCurrentContext, { "OSMesaGetCurrentContext", NULL } },
{ (PRFuncPtr*) &fMakeCurrent, { "OSMesaMakeCurrent", NULL } },
{ (PRFuncPtr*) &fGetProcAddress, { "OSMesaGetProcAddress", NULL } },
{ NULL, { NULL } }
};
if (!LibrarySymbolLoader::LoadSymbols(mOSMesaLibrary, &symbols[0])) {
NS_WARNING("Couldn't find required entry points in OSMesa libary");
return PR_FALSE;
}
mInitialized = PR_TRUE;
return PR_TRUE;
}
class GLContextOSMesa : public GLContext
{
public:
GLContextOSMesa()
: mThebesSurface(nsnull),
mContext(nsnull)
{
}
~GLContextOSMesa()
{
if (mContext)
sOSMesaLibrary.fDestroyContext(mContext);
}
PRBool Init(const gfxIntSize &aSize, const GLContextProvider::ContextFormat& aFormat)
{
int osmesa_format = -1;
int gfxasurface_imageformat = -1;
PRBool format_accepted = PR_FALSE;
if (aFormat.red == 8 && aFormat.green == 8 && aFormat.blue == 8) {
if (aFormat.alpha == 8) {
osmesa_format = OSMESA_BGRA;
gfxasurface_imageformat = gfxASurface::ImageFormatARGB32;
format_accepted = PR_TRUE;
}
else if (aFormat.alpha == 0) {
// we can't use OSMESA_BGR because it is packed 24 bits per pixel.
// So we use OSMESA_BGRA and have to use ImageFormatRGB24
// to make sure that the dummy alpha channel is ignored.
osmesa_format = OSMESA_BGRA;
gfxasurface_imageformat = gfxASurface::ImageFormatRGB24;
format_accepted = PR_TRUE;
}
}
if (!format_accepted) {
NS_WARNING("Pixel format not supported with OSMesa.");
return PR_FALSE;
}
mThebesSurface = new gfxImageSurface(aSize, gfxASurface::gfxImageFormat(gfxasurface_imageformat));
if (mThebesSurface->CairoStatus() != 0) {
NS_WARNING("image surface failed");
return PR_FALSE;
}
mContext = sOSMesaLibrary.fCreateContextExt(osmesa_format, aFormat.depth, aFormat.stencil, 0, NULL);
if (!mContext) {
NS_WARNING("OSMesaCreateContextExt failed!");
return PR_FALSE;
}
if (!MakeCurrent()) return PR_FALSE;
if (!SetupLookupFunction()) return PR_FALSE;
return InitWithPrefix("gl", PR_TRUE);
}
PRBool MakeCurrent()
{
PRBool succeeded
= sOSMesaLibrary.fMakeCurrent (mContext, mThebesSurface->Data(),
LOCAL_GL_UNSIGNED_BYTE,
mThebesSurface->Width(),
mThebesSurface->Height());
NS_ASSERTION(succeeded, "Failed to make OSMesa context current!");
return succeeded;
}
PRBool SetupLookupFunction()
{
mLookupFunc = (PlatformLookupFunction)sOSMesaLibrary.fGetProcAddress;
return PR_TRUE;
}
void *GetNativeData(NativeDataType aType)
{
switch (aType) {
case NativeImageSurface:
return mThebesSurface.get();
default:
return nsnull;
}
}
private:
nsRefPtr<gfxImageSurface> mThebesSurface;
PrivateOSMesaContext mContext;
};
already_AddRefed<GLContext>
GLContextProviderOSMesa::CreateForWindow(nsIWidget *aWidget)
{
return nsnull;
}
already_AddRefed<GLContext>
GLContextProviderOSMesa::CreatePBuffer(const gfxIntSize &aSize, const ContextFormat& aFormat)
{
if (!sOSMesaLibrary.EnsureInitialized()) {
return nsnull;
}
nsRefPtr<GLContextOSMesa> glContext = new GLContextOSMesa;
if (!glContext->Init(aSize, aFormat)) {
return nsnull;
}
return glContext.forget().get();
}
} /* namespace gl */
} /* namespace mozilla */

View File

@ -34,6 +34,7 @@ CPPSRCS = \
gfxUserFontSet.cpp \ gfxUserFontSet.cpp \
gfxUtils.cpp \ gfxUtils.cpp \
GLContext.cpp \ GLContext.cpp \
GLContextProviderOSMesa.cpp \
$(NULL) $(NULL)
ifdef MOZ_IPC ifdef MOZ_IPC