| /* |
| * Copyright 2012 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "SkSurface_Base.h" |
| #include "SkImagePriv.h" |
| #include "SkCanvas.h" |
| #include "SkGpuDevice.h" |
| |
| class SkSurface_Gpu : public SkSurface_Base { |
| public: |
| SK_DECLARE_INST_COUNT(SkSurface_Gpu) |
| |
| SkSurface_Gpu(GrRenderTarget*, bool cached, TextRenderMode trm); |
| virtual ~SkSurface_Gpu(); |
| |
| virtual SkCanvas* onNewCanvas() SK_OVERRIDE; |
| virtual SkSurface* onNewSurface(const SkImageInfo&) SK_OVERRIDE; |
| virtual SkImage* onNewImageSnapshot() SK_OVERRIDE; |
| virtual void onDraw(SkCanvas*, SkScalar x, SkScalar y, |
| const SkPaint*) SK_OVERRIDE; |
| virtual void onCopyOnWrite(ContentChangeMode) SK_OVERRIDE; |
| virtual void onDiscard() SK_OVERRIDE; |
| |
| private: |
| SkGpuDevice* fDevice; |
| |
| typedef SkSurface_Base INHERITED; |
| }; |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkSurface_Gpu::SkSurface_Gpu(GrRenderTarget* renderTarget, bool cached, TextRenderMode trm) |
| : INHERITED(renderTarget->width(), renderTarget->height()) { |
| int flags = 0; |
| flags |= cached ? SkGpuDevice::kCached_Flag : 0; |
| flags |= (kDistanceField_TextRenderMode == trm) ? SkGpuDevice::kDFFonts_Flag : 0; |
| fDevice = SkGpuDevice::Create(renderTarget, flags); |
| |
| if (kRGB_565_GrPixelConfig != renderTarget->config()) { |
| fDevice->clear(0x0); |
| } |
| } |
| |
| SkSurface_Gpu::~SkSurface_Gpu() { |
| SkSafeUnref(fDevice); |
| } |
| |
| SkCanvas* SkSurface_Gpu::onNewCanvas() { |
| return SkNEW_ARGS(SkCanvas, (fDevice)); |
| } |
| |
| SkSurface* SkSurface_Gpu::onNewSurface(const SkImageInfo& info) { |
| GrRenderTarget* rt = fDevice->accessRenderTarget(); |
| int sampleCount = rt->numSamples(); |
| return SkSurface::NewRenderTarget(fDevice->context(), info, sampleCount); |
| } |
| |
| SkImage* SkSurface_Gpu::onNewImageSnapshot() { |
| return SkImage::NewTexture(fDevice->accessBitmap(false)); |
| } |
| |
| void SkSurface_Gpu::onDraw(SkCanvas* canvas, SkScalar x, SkScalar y, |
| const SkPaint* paint) { |
| canvas->drawBitmap(fDevice->accessBitmap(false), x, y, paint); |
| } |
| |
| // Create a new SkGpuDevice and, if necessary, copy the contents of the old |
| // device into it. Note that this flushes the SkGpuDevice but |
| // doesn't force an OpenGL flush. |
| void SkSurface_Gpu::onCopyOnWrite(ContentChangeMode mode) { |
| GrRenderTarget* rt = fDevice->accessRenderTarget(); |
| // are we sharing our render target with the image? |
| SkASSERT(NULL != this->getCachedImage()); |
| if (rt->asTexture() == SkTextureImageGetTexture(this->getCachedImage())) { |
| // We call createCompatibleDevice because it uses the texture cache. This isn't |
| // necessarily correct (http://skbug.com/2252), but never using the cache causes |
| // a Chromium regression. (http://crbug.com/344020) |
| SkGpuDevice* newDevice = static_cast<SkGpuDevice*>( |
| fDevice->createCompatibleDevice(fDevice->imageInfo())); |
| SkAutoTUnref<SkGpuDevice> aurd(newDevice); |
| if (kRetain_ContentChangeMode == mode) { |
| fDevice->context()->copyTexture(rt->asTexture(), newDevice->accessRenderTarget()); |
| } |
| SkASSERT(NULL != this->getCachedCanvas()); |
| SkASSERT(this->getCachedCanvas()->getDevice() == fDevice); |
| |
| this->getCachedCanvas()->setRootDevice(newDevice); |
| SkRefCnt_SafeAssign(fDevice, newDevice); |
| } else if (kDiscard_ContentChangeMode == mode) { |
| this->SkSurface_Gpu::onDiscard(); |
| } |
| } |
| |
| void SkSurface_Gpu::onDiscard() { |
| fDevice->accessRenderTarget()->discard(); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////// |
| |
| SkSurface* SkSurface::NewRenderTargetDirect(GrRenderTarget* target, TextRenderMode trm) { |
| if (NULL == target) { |
| return NULL; |
| } |
| return SkNEW_ARGS(SkSurface_Gpu, (target, false, trm)); |
| } |
| |
| SkSurface* SkSurface::NewRenderTarget(GrContext* ctx, const SkImageInfo& info, int sampleCount, |
| TextRenderMode trm) { |
| if (NULL == ctx) { |
| return NULL; |
| } |
| |
| GrTextureDesc desc; |
| desc.fFlags = kRenderTarget_GrTextureFlagBit | kCheckAllocation_GrTextureFlagBit; |
| desc.fWidth = info.width(); |
| desc.fHeight = info.height(); |
| desc.fConfig = SkImageInfo2GrPixelConfig(info); |
| desc.fSampleCnt = sampleCount; |
| |
| SkAutoTUnref<GrTexture> tex(ctx->createUncachedTexture(desc, NULL, 0)); |
| if (NULL == tex) { |
| return NULL; |
| } |
| |
| return SkNEW_ARGS(SkSurface_Gpu, (tex->asRenderTarget(), false, trm)); |
| } |
| |
| SkSurface* SkSurface::NewScratchRenderTarget(GrContext* ctx, const SkImageInfo& info, |
| int sampleCount, TextRenderMode trm) { |
| if (NULL == ctx) { |
| return NULL; |
| } |
| |
| GrTextureDesc desc; |
| desc.fFlags = kRenderTarget_GrTextureFlagBit | kCheckAllocation_GrTextureFlagBit; |
| desc.fWidth = info.width(); |
| desc.fHeight = info.height(); |
| desc.fConfig = SkImageInfo2GrPixelConfig(info); |
| desc.fSampleCnt = sampleCount; |
| |
| SkAutoTUnref<GrTexture> tex(ctx->lockAndRefScratchTexture(desc, GrContext::kExact_ScratchTexMatch)); |
| |
| if (NULL == tex) { |
| return NULL; |
| } |
| |
| return SkNEW_ARGS(SkSurface_Gpu, (tex->asRenderTarget(), true, trm)); |
| } |