2012-05-23 11:43:41 -07:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef GrContextFactory_DEFINED
|
|
|
|
#define GrContextFactory_DEFINED
|
|
|
|
|
|
|
|
#if SK_ANGLE
|
|
|
|
#include "gl/SkANGLEGLContext.h"
|
|
|
|
#endif
|
|
|
|
#include "gl/SkDebugGLContext.h"
|
|
|
|
#if SK_MESA
|
|
|
|
#include "gl/SkMesaGLContext.h"
|
|
|
|
#endif
|
|
|
|
#include "gl/SkNativeGLContext.h"
|
|
|
|
#include "gl/SkNullGLContext.h"
|
|
|
|
|
|
|
|
#include "GrContext.h"
|
2012-08-23 11:34:29 -07:00
|
|
|
#include "SkTArray.h"
|
2012-05-23 11:43:41 -07:00
|
|
|
|
|
|
|
/**
|
2012-08-23 11:34:29 -07:00
|
|
|
* This is a simple class that is useful in test apps that use different
|
2012-05-23 11:43:41 -07:00
|
|
|
* GrContexts backed by different types of GL contexts. It manages creating the
|
|
|
|
* GL context and a GrContext that uses it. The GL/Gr contexts persist until the
|
|
|
|
* factory is destroyed (though the caller can always grab a ref on the returned
|
|
|
|
* GrContext to make it outlive the factory).
|
|
|
|
*/
|
2013-04-02 13:49:07 -07:00
|
|
|
class GrContextFactory : GrNoncopyable {
|
2012-05-23 11:43:41 -07:00
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Types of GL contexts supported.
|
|
|
|
*/
|
|
|
|
enum GLContextType {
|
|
|
|
kNative_GLContextType,
|
|
|
|
#if SK_ANGLE
|
|
|
|
kANGLE_GLContextType,
|
|
|
|
#endif
|
|
|
|
#if SK_MESA
|
|
|
|
kMESA_GLContextType,
|
|
|
|
#endif
|
|
|
|
kNull_GLContextType,
|
|
|
|
kDebug_GLContextType,
|
2013-04-02 13:49:07 -07:00
|
|
|
|
|
|
|
kLastGLContextType = kDebug_GLContextType
|
2012-05-23 11:43:41 -07:00
|
|
|
};
|
|
|
|
|
2013-04-02 13:49:07 -07:00
|
|
|
static const int kGLContextTypeCnt = kLastGLContextType + 1;
|
|
|
|
|
|
|
|
static bool IsRenderingGLContext(GLContextType type) {
|
|
|
|
switch (type) {
|
|
|
|
case kNull_GLContextType:
|
|
|
|
case kDebug_GLContextType:
|
|
|
|
return false;
|
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char* GLContextTypeName(GLContextType type) {
|
|
|
|
switch (type) {
|
|
|
|
case kNative_GLContextType:
|
|
|
|
return "native";
|
|
|
|
case kNull_GLContextType:
|
|
|
|
return "null";
|
|
|
|
#if SK_ANGLE
|
|
|
|
case kANGLE_GLContextType:
|
|
|
|
return "angle";
|
|
|
|
#endif
|
|
|
|
#if SK_MESA
|
|
|
|
case kMESA_GLContextType:
|
|
|
|
return "mesa";
|
|
|
|
#endif
|
|
|
|
case kDebug_GLContextType:
|
|
|
|
return "debug";
|
|
|
|
default:
|
|
|
|
GrCrash("Unknown GL Context type.");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-05-23 11:43:41 -07:00
|
|
|
GrContextFactory() {
|
|
|
|
}
|
|
|
|
|
2013-04-02 13:49:07 -07:00
|
|
|
~GrContextFactory() { this->destroyContexts(); }
|
|
|
|
|
|
|
|
void destroyContexts() {
|
2012-05-23 11:43:41 -07:00
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
|
|
|
fContexts[i].fGrContext->unref();
|
|
|
|
fContexts[i].fGLContext->unref();
|
|
|
|
}
|
2013-04-02 13:49:07 -07:00
|
|
|
fContexts.reset();
|
2012-05-23 11:43:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-04-02 13:49:07 -07:00
|
|
|
* Get a GrContext initialized with a type of GL context. It also makes the GL context current.
|
2012-05-23 11:43:41 -07:00
|
|
|
*/
|
|
|
|
GrContext* get(GLContextType type) {
|
|
|
|
|
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
|
|
|
if (fContexts[i].fType == type) {
|
2013-04-02 13:49:07 -07:00
|
|
|
fContexts[i].fGLContext->makeCurrent();
|
2012-05-23 11:43:41 -07:00
|
|
|
return fContexts[i].fGrContext;
|
|
|
|
}
|
|
|
|
}
|
2013-04-02 13:49:07 -07:00
|
|
|
SkAutoTUnref<SkGLContextHelper> glCtx;
|
2012-05-23 11:43:41 -07:00
|
|
|
SkAutoTUnref<GrContext> grCtx;
|
|
|
|
switch (type) {
|
|
|
|
case kNative_GLContextType:
|
2012-08-23 11:34:29 -07:00
|
|
|
glCtx.reset(SkNEW(SkNativeGLContext));
|
2012-05-23 11:43:41 -07:00
|
|
|
break;
|
|
|
|
#ifdef SK_ANGLE
|
|
|
|
case kANGLE_GLContextType:
|
2012-08-23 11:34:29 -07:00
|
|
|
glCtx.reset(SkNEW(SkANGLEGLContext));
|
2012-05-23 11:43:41 -07:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
#ifdef SK_MESA
|
|
|
|
case kMESA_GLContextType:
|
2012-08-23 11:34:29 -07:00
|
|
|
glCtx.reset(SkNEW(SkMesaGLContext));
|
2012-05-23 11:43:41 -07:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
case kNull_GLContextType:
|
2012-08-23 11:34:29 -07:00
|
|
|
glCtx.reset(SkNEW(SkNullGLContext));
|
2012-05-23 11:43:41 -07:00
|
|
|
break;
|
|
|
|
case kDebug_GLContextType:
|
2012-08-23 11:34:29 -07:00
|
|
|
glCtx.reset(SkNEW(SkDebugGLContext));
|
2012-05-23 11:43:41 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
static const int kBogusSize = 1;
|
|
|
|
if (!glCtx.get()) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (!glCtx.get()->init(kBogusSize, kBogusSize)) {
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-04-02 13:49:07 -07:00
|
|
|
GrBackendContext p3dctx = reinterpret_cast<GrBackendContext>(glCtx.get()->gl());
|
|
|
|
grCtx.reset(GrContext::Create(kOpenGL_GrBackend, p3dctx));
|
2012-05-23 11:43:41 -07:00
|
|
|
if (!grCtx.get()) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
GPUContext& ctx = fContexts.push_back();
|
|
|
|
ctx.fGLContext = glCtx.get();
|
|
|
|
ctx.fGLContext->ref();
|
|
|
|
ctx.fGrContext = grCtx.get();
|
|
|
|
ctx.fGrContext->ref();
|
|
|
|
ctx.fType = type;
|
|
|
|
return ctx.fGrContext;
|
|
|
|
}
|
2012-08-23 11:34:29 -07:00
|
|
|
|
|
|
|
// Returns the GLContext of the given type. If it has not been created yet,
|
|
|
|
// NULL is returned instead.
|
2013-04-02 13:49:07 -07:00
|
|
|
SkGLContextHelper* getGLContext(GLContextType type) {
|
2012-08-23 11:34:29 -07:00
|
|
|
for (int i = 0; i < fContexts.count(); ++i) {
|
|
|
|
if (fContexts[i].fType == type) {
|
|
|
|
return fContexts[i].fGLContext;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2012-05-23 11:43:41 -07:00
|
|
|
private:
|
|
|
|
struct GPUContext {
|
|
|
|
GLContextType fType;
|
2013-04-02 13:49:07 -07:00
|
|
|
SkGLContextHelper* fGLContext;
|
2012-05-23 11:43:41 -07:00
|
|
|
GrContext* fGrContext;
|
|
|
|
};
|
|
|
|
SkTArray<GPUContext, true> fContexts;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|