SkTDArray<CleanUpData> fCleanUpData;
- GrContext(GrGpu* gpu);
+ GrContext(); // init must be called after the constructor.
+ bool init(GrBackend, GrBackendContext);
void setupDrawBuffer();
void GrClipMaskManager::releaseResources() {
fAACache.releaseResources();
}
+
+void GrClipMaskManager::setGpu(GrGpu* gpu) {
+ fGpu = gpu;
+ fAACache.setContext(gpu->getContext());
+}
}
}
- void setContext(GrContext* context) {
- fAACache.setContext(context);
- }
-
GrContext* getContext() {
return fAACache.getContext();
}
- void setGpu(GrGpu* gpu) {
- fGpu = gpu;
- }
-
+ void setGpu(GrGpu* gpu);
private:
/**
* Informs the helper function adjustStencilParams() about how the stencil
#define ASSERT_OWNED_RESOURCE(R) GrAssert(!(R) || (R)->getContext() == this)
-GrContext* GrContext::Create(GrBackend backend, GrBackendContext context) {
- GrContext* ctx = NULL;
- GrGpu* fGpu = GrGpu::Create(backend, context);
- if (NULL != fGpu) {
- ctx = SkNEW_ARGS(GrContext, (fGpu));
- fGpu->unref();
+GrContext* GrContext::Create(GrBackend backend, GrBackendContext backendContext) {
+ GrContext* context = SkNEW(GrContext);
+ if (context->init(backend, backendContext)) {
+ return context;
+ } else {
+ context->unref();
+ return NULL;
}
- return ctx;
}
namespace {
void DeleteThreadInstanceCount(void* v) {
delete reinterpret_cast<int*>(v);
}
-#define THREAD_INSTANCE_COUNT \
- (*reinterpret_cast<int*>(SkTLS::Get(CreateThreadInstanceCount, \
- DeleteThreadInstanceCount)))
+#define THREAD_INSTANCE_COUNT \
+ (*reinterpret_cast<int*>(SkTLS::Get(CreateThreadInstanceCount, DeleteThreadInstanceCount)))
+}
+
+GrContext::GrContext() {
+ ++THREAD_INSTANCE_COUNT;
+ fDrawState = NULL;
+ fGpu = NULL;
+ fPathRendererChain = NULL;
+ fSoftwarePathRenderer = NULL;
+ fTextureCache = NULL;
+ fFontCache = NULL;
+ fDrawBuffer = NULL;
+ fDrawBufferVBAllocPool = NULL;
+ fDrawBufferIBAllocPool = NULL;
+ fAARectRenderer = NULL;
+}
+
+bool GrContext::init(GrBackend backend, GrBackendContext backendContext) {
+ GrAssert(NULL == fGpu);
+
+ fGpu = GrGpu::Create(backend, backendContext, this);
+ if (NULL == fGpu) {
+ return false;
+ }
+
+ fDrawState = SkNEW(GrDrawState);
+ fGpu->setDrawState(fDrawState);
+
+
+ fTextureCache = SkNEW_ARGS(GrResourceCache,
+ (MAX_TEXTURE_CACHE_COUNT,
+ MAX_TEXTURE_CACHE_BYTES));
+ fFontCache = SkNEW_ARGS(GrFontCache, (fGpu));
+
+ fLastDrawWasBuffered = kNo_BufferedDraw;
+
+ fAARectRenderer = SkNEW(GrAARectRenderer);
+
+ fDidTestPMConversions = false;
+
+ this->setupDrawBuffer();
+ return true;
}
int GrContext::GetThreadInstanceCount() {
}
void GrContext::contextLost() {
- contextDestroyed();
+ this->contextDestroyed();
this->setupDrawBuffer();
}
}
void GrContext::flushDrawBuffer() {
- if (fDrawBuffer) {
- // With addition of the AA clip path, flushing the draw buffer can
- // result in the generation of an AA clip mask. During this
- // process the SW path renderer may be invoked which recusively
- // calls this method (via internalWriteTexturePixels) creating
- // infinite recursion
- GrInOrderDrawBuffer* temp = fDrawBuffer;
- fDrawBuffer = NULL;
-
- temp->flushTo(fGpu);
-
- fDrawBuffer = temp;
+ if (NULL != fDrawBuffer && !fDrawBuffer->isFlushing()) {
+ fDrawBuffer->flush();
}
}
return bits;
}
-GrContext::GrContext(GrGpu* gpu) {
- ++THREAD_INSTANCE_COUNT;
-
- fGpu = gpu;
- fGpu->ref();
- fGpu->setContext(this);
-
- fDrawState = SkNEW(GrDrawState);
- fGpu->setDrawState(fDrawState);
-
- fPathRendererChain = NULL;
- fSoftwarePathRenderer = NULL;
-
- fTextureCache = SkNEW_ARGS(GrResourceCache,
- (MAX_TEXTURE_CACHE_COUNT,
- MAX_TEXTURE_CACHE_BYTES));
- fFontCache = SkNEW_ARGS(GrFontCache, (fGpu));
-
- fLastDrawWasBuffered = kNo_BufferedDraw;
-
- fDrawBuffer = NULL;
- fDrawBufferVBAllocPool = NULL;
- fDrawBufferIBAllocPool = NULL;
-
- fAARectRenderer = SkNEW(GrAARectRenderer);
-
- fDidTestPMConversions = false;
-
- this->setupDrawBuffer();
-}
-
void GrContext::setupDrawBuffer() {
GrAssert(NULL == fDrawBuffer);
DRAW_BUFFER_IBPOOL_PREALLOC_BUFFERS));
fDrawBuffer = SkNEW_ARGS(GrInOrderDrawBuffer, (fGpu,
- fDrawBufferVBAllocPool,
- fDrawBufferIBAllocPool));
+ fDrawBufferVBAllocPool,
+ fDrawBufferIBAllocPool));
- if (fDrawBuffer) {
- fDrawBuffer->setAutoFlushTarget(fGpu);
- fDrawBuffer->setDrawState(fDrawState);
- }
+ fDrawBuffer->setDrawState(fDrawState);
}
GrDrawTarget* GrContext::getTextTarget(const GrPaint& paint) {
#define DEBUG_INVAL_BUFFER 0xdeadcafe
#define DEBUG_INVAL_START_IDX -1
-GrDrawTarget::GrDrawTarget() : fClip(NULL) {
+GrDrawTarget::GrDrawTarget(GrContext* context)
+ : fClip(NULL)
+ , fContext(context) {
+ GrAssert(NULL != context);
+
fDrawState = &fDefaultDrawState;
// We assume that fDrawState always owns a ref to the object it points at.
fDefaultDrawState.ref();
///////////////////////////////////////////////////////////////////////////
- GrDrawTarget();
+ // The context may not be fully constructed and should not be used during GrDrawTarget
+ // construction.
+ GrDrawTarget(GrContext* context);
virtual ~GrDrawTarget();
/**
}
}
+ GrContext* getContext() { return fContext; }
+ const GrContext* getContext() const { return fContext; }
+
// allows derived class to set the caps
CapsInternals* capsInternals() { return &fCaps.fInternals; }
const GrClipData* fClip;
GrDrawState* fDrawState;
GrDrawState fDefaultDrawState;
+ // The context owns us, not vice-versa, so this ptr is not ref'ed by DrawTarget.
+ GrContext* fContext;
typedef GrRefCnt INHERITED;
};
#define DEBUG_INVAL_BUFFER 0xdeadcafe
#define DEBUG_INVAL_START_IDX -1
-GrGpu::GrGpu()
- : fContext(NULL)
+GrGpu::GrGpu(GrContext* context)
+ : GrDrawTarget(context)
, fResetTimestamp(kExpiredTimestamp+1)
, fVertexPool(NULL)
, fIndexPool(NULL)
/**
* Create an instance of GrGpu that matches the specified backend. If the requested backend is
- * not supported (at compile-time or run-time) this returns NULL.
+ * not supported (at compile-time or run-time) this returns NULL. The context will not be
+ * fully constructed and should not be used by GrGpu until after this function returns.
*/
- static GrGpu* Create(GrBackend, GrBackendContext);
+ static GrGpu* Create(GrBackend, GrBackendContext, GrContext* context);
////////////////////////////////////////////////////////////////////////////
- GrGpu();
+ GrGpu(GrContext* context);
virtual ~GrGpu();
- // The GrContext sets itself as the owner of this Gpu object
- void setContext(GrContext* context) {
- GrAssert(NULL == fContext);
- fContext = context;
- fClipMaskManager.setContext(context);
- }
- GrContext* getContext() { return fContext; }
- const GrContext* getContext() const { return fContext; }
+ GrContext* getContext() { return this->INHERITED::getContext(); }
+ const GrContext* getContext() const { return this->INHERITED::getContext(); }
/**
* The GrGpu object normally assumes that no outsider is setting state
};
typedef SkTInternalLList<GrResource> ResourceList;
SkSTArray<kPreallocGeomPoolStateStackCnt, GeometryPoolState, true> fGeomPoolStateStack;
- GrContext* fContext; // not reffed
ResetTimestamp fResetTimestamp;
GrVertexBufferAllocPool* fVertexPool;
GrIndexBufferAllocPool* fIndexPool;
#include "GrGpu.h"
#include "gl/GrGpuGL.h"
-GrGpu* GrGpu::Create(GrBackend backend, GrBackendContext context) {
+GrGpu* GrGpu::Create(GrBackend backend, GrBackendContext backendContext, GrContext* context) {
const GrGLInterface* glInterface = NULL;
SkAutoTUnref<const GrGLInterface> glInterfaceUnref;
if (kOpenGL_GrBackend == backend) {
- glInterface = reinterpret_cast<const GrGLInterface*>(context);
+ glInterface = reinterpret_cast<const GrGLInterface*>(backendContext);
if (NULL == glInterface) {
glInterface = GrGLDefaultInterface();
// By calling GrGLDefaultInterface we've taken a ref on the
}
GrGLContextInfo ctxInfo(glInterface);
if (ctxInfo.isInitialized()) {
- return SkNEW_ARGS(GrGpuGL, (ctxInfo));
+ return SkNEW_ARGS(GrGpuGL, (ctxInfo, context));
}
}
return NULL;
#include "GrIndexBuffer.h"
#include "GrPath.h"
#include "GrRenderTarget.h"
+#include "GrTemplates.h"
#include "GrTexture.h"
#include "GrVertexBuffer.h"
-GrInOrderDrawBuffer::GrInOrderDrawBuffer(const GrGpu* gpu,
+GrInOrderDrawBuffer::GrInOrderDrawBuffer(GrGpu* gpu,
GrVertexBufferAllocPool* vertexPool,
GrIndexBufferAllocPool* indexPool)
- : fAutoFlushTarget(NULL)
+ : GrDrawTarget(gpu->getContext())
+ , fDstGpu(gpu)
, fClipSet(true)
, fClipProxyState(kUnknown_ClipProxyState)
, fVertexPool(*vertexPool)
, fIndexPool(*indexPool)
, fFlushing(false) {
- fGpu.reset(SkRef(gpu));
- fCaps = gpu->getCaps();
+ fDstGpu->ref();
+ fCaps = fDstGpu->getCaps();
GrAssert(NULL != vertexPool);
GrAssert(NULL != indexPool);
this->reset();
// This must be called by before the GrDrawTarget destructor
this->releaseGeometry();
- GrSafeUnref(fAutoFlushTarget);
+ fDstGpu->unref();
}
////////////////////////////////////////////////////////////////////////////////
}
}
- this->setIndexSourceToBuffer(fGpu->getQuadIndexBuffer());
+ this->setIndexSourceToBuffer(this->getContext()->getQuadIndexBuffer());
this->drawIndexedInstances(kTriangles_GrPrimitiveType, 1, 4, 6, &devBounds);
}
fClipSet = true;
}
-bool GrInOrderDrawBuffer::flushTo(GrDrawTarget* target) {
+bool GrInOrderDrawBuffer::flush() {
GrAssert(kReserved_GeometrySrcType != this->getGeomSrc().fVertexSrc);
GrAssert(kReserved_GeometrySrcType != this->getGeomSrc().fIndexSrc);
- GrAssert(NULL != target);
- GrAssert(target != this); // not considered and why?
-
int numCmds = fCmds.count();
if (0 == numCmds) {
return false;
}
+ GrAssert(!fFlushing);
+
+ GrAutoTRestore<bool> flushRestore(&fFlushing);
+ fFlushing = true;
fVertexPool.unlock();
fIndexPool.unlock();
- GrDrawTarget::AutoClipRestore acr(target);
- AutoGeometryAndStatePush agasp(target, kPreserve_ASRInit);
+ GrDrawTarget::AutoClipRestore acr(fDstGpu);
+ AutoGeometryAndStatePush agasp(fDstGpu, kPreserve_ASRInit);
GrDrawState playbackState;
- GrDrawState* prevDrawState = target->drawState();
+ GrDrawState* prevDrawState = fDstGpu->drawState();
prevDrawState->ref();
- target->setDrawState(&playbackState);
+ fDstGpu->setDrawState(&playbackState);
GrClipData clipData;
int currDraw = 0;
int currStencilPath = 0;
-
for (int c = 0; c < numCmds; ++c) {
switch (fCmds[c]) {
case kDraw_Cmd: {
const DrawRecord& draw = fDraws[currDraw];
- target->setVertexSourceToBuffer(draw.fVertexBuffer);
+ fDstGpu->setVertexSourceToBuffer(draw.fVertexBuffer);
if (draw.isIndexed()) {
- target->setIndexSourceToBuffer(draw.fIndexBuffer);
+ fDstGpu->setIndexSourceToBuffer(draw.fIndexBuffer);
}
- target->executeDraw(draw);
+ fDstGpu->executeDraw(draw);
++currDraw;
break;
}
case kStencilPath_Cmd: {
const StencilPath& sp = fStencilPaths[currStencilPath];
- target->stencilPath(sp.fPath.get(), sp.fStroke, sp.fFill);
+ fDstGpu->stencilPath(sp.fPath.get(), sp.fStroke, sp.fFill);
++currStencilPath;
break;
}
case kSetClip_Cmd:
clipData.fClipStack = &fClips[currClip];
clipData.fOrigin = fClipOrigins[currClip];
- target->setClip(&clipData);
+ fDstGpu->setClip(&clipData);
++currClip;
break;
case kClear_Cmd:
- target->clear(&fClears[currClear].fRect,
- fClears[currClear].fColor,
- fClears[currClear].fRenderTarget);
+ fDstGpu->clear(&fClears[currClear].fRect,
+ fClears[currClear].fColor,
+ fClears[currClear].fRenderTarget);
++currClear;
break;
}
GrAssert(fClears.count() == currClear);
GrAssert(fDraws.count() == currDraw);
- target->setDrawState(prevDrawState);
+ fDstGpu->setDrawState(prevDrawState);
prevDrawState->unref();
this->reset();
return true;
}
-void GrInOrderDrawBuffer::setAutoFlushTarget(GrDrawTarget* target) {
- GrSafeAssign(fAutoFlushTarget, target);
-}
-
void GrInOrderDrawBuffer::willReserveVertexAndIndexSpace(
int vertexCount,
int indexCount) {
- if (NULL != fAutoFlushTarget) {
- // We use geometryHints() to know whether to flush the draw buffer. We
- // can't flush if we are inside an unbalanced pushGeometrySource.
- // Moreover, flushing blows away vertex and index data that was
- // previously reserved. So if the vertex or index data is pulled from
- // reserved space and won't be released by this request then we can't
- // flush.
- bool insideGeoPush = fGeoPoolStateStack.count() > 1;
-
- bool unreleasedVertexSpace =
- !vertexCount &&
- kReserved_GeometrySrcType == this->getGeomSrc().fVertexSrc;
-
- bool unreleasedIndexSpace =
- !indexCount &&
- kReserved_GeometrySrcType == this->getGeomSrc().fIndexSrc;
-
- // we don't want to finalize any reserved geom on the target since
- // we don't know that the client has finished writing to it.
- bool targetHasReservedGeom =
- fAutoFlushTarget->hasReservedVerticesOrIndices();
-
- int vcount = vertexCount;
- int icount = indexCount;
-
- if (!insideGeoPush &&
- !unreleasedVertexSpace &&
- !unreleasedIndexSpace &&
- !targetHasReservedGeom &&
- this->geometryHints(&vcount, &icount)) {
-
- this->flushTo(fAutoFlushTarget);
- }
+ // We use geometryHints() to know whether to flush the draw buffer. We
+ // can't flush if we are inside an unbalanced pushGeometrySource.
+ // Moreover, flushing blows away vertex and index data that was
+ // previously reserved. So if the vertex or index data is pulled from
+ // reserved space and won't be released by this request then we can't
+ // flush.
+ bool insideGeoPush = fGeoPoolStateStack.count() > 1;
+
+ bool unreleasedVertexSpace =
+ !vertexCount &&
+ kReserved_GeometrySrcType == this->getGeomSrc().fVertexSrc;
+
+ bool unreleasedIndexSpace =
+ !indexCount &&
+ kReserved_GeometrySrcType == this->getGeomSrc().fIndexSrc;
+
+ // we don't want to finalize any reserved geom on the target since
+ // we don't know that the client has finished writing to it.
+ bool targetHasReservedGeom = fDstGpu->hasReservedVerticesOrIndices();
+
+ int vcount = vertexCount;
+ int icount = indexCount;
+
+ if (!insideGeoPush &&
+ !unreleasedVertexSpace &&
+ !unreleasedIndexSpace &&
+ !targetHasReservedGeom &&
+ this->geometryHints(&vcount, &icount)) {
+
+ this->flush();
}
}
/**
* Creates a GrInOrderDrawBuffer
*
- * @param gpu the gpu object where this will be played back
- * (possible indirectly). GrResources used with the draw
- * buffer are created by this gpu object.
+ * @param gpu the gpu object that this draw buffer flushes to.
* @param vertexPool pool where vertices for queued draws will be saved when
* the vertex source is either reserved or array.
* @param indexPool pool where indices for queued draws will be saved when
* the index source is either reserved or array.
*/
- GrInOrderDrawBuffer(const GrGpu* gpu,
+ GrInOrderDrawBuffer(GrGpu* gpu,
GrVertexBufferAllocPool* vertexPool,
GrIndexBufferAllocPool* indexPool);
void reset();
/**
- * This plays the queued up draws to another target. It also resets this object (i.e. flushing
+ * This plays the queued up draws to its GrGpu target. It also resets this object (i.e. flushing
* is destructive). This buffer must not have an active reserved vertex or index source. Any
* reserved geometry on the target will be finalized because it's geometry source will be pushed
* before flushing and popped afterwards.
*
* @return false if the playback trivially drew nothing because nothing was recorded.
- *
- * @param target the target to receive the playback
*/
- bool flushTo(GrDrawTarget* target);
+ bool flush();
- /**
- * This function allows the draw buffer to automatically flush itself to another target. This
- * means the buffer may internally call this->flushTo(target) when it is safe to do so.
- *
- * When the auto flush target is set to NULL (as it initially is) the draw buffer will never
- * automatically flush itself.
- */
- void setAutoFlushTarget(GrDrawTarget* target);
+ bool isFlushing() const { return fFlushing; }
// overrides from GrDrawTarget
virtual bool geometryHints(int* vertexCount,
kGeoPoolStatePreAllocCnt = 4,
};
- SkAutoTUnref<const GrGpu> fGpu;
-
SkSTArray<kCmdPreallocCnt, uint8_t, true> fCmds;
GrSTAllocator<kDrawPreallocCnt, DrawRecord> fDraws;
GrSTAllocator<kStatePreallocCnt, StencilPath> fStencilPaths;
GrSTAllocator<kClipPreallocCnt, SkClipStack> fClips;
GrSTAllocator<kClipPreallocCnt, SkIPoint> fClipOrigins;
- GrDrawTarget* fAutoFlushTarget;
+ GrDrawTarget* fDstGpu;
bool fClipSet;
return status == GR_GL_FRAMEBUFFER_COMPLETE;
}
-GrGpuGL::GrGpuGL(const GrGLContextInfo& ctxInfo) : fGLContextInfo(ctxInfo) {
+GrGpuGL::GrGpuGL(const GrGLContextInfo& ctxInfo, GrContext* context)
+ : GrGpu(context)
+ , fGLContextInfo(ctxInfo) {
GrAssert(ctxInfo.isInitialized());
class GrGpuGL : public GrGpu {
public:
- GrGpuGL(const GrGLContextInfo& ctxInfo);
+ GrGpuGL(const GrGLContextInfo& ctxInfo, GrContext* context);
virtual ~GrGpuGL();
const GrGLInterface* glInterface() const {