reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2012 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 8 | #include "SkSurface_Gpu.h" |
| 9 | |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 10 | #include "SkCanvas.h" |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 11 | #include "SkGpuDevice.h" |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 12 | #include "SkImage_Base.h" |
| 13 | #include "SkImagePriv.h" |
| 14 | #include "SkSurface_Base.h" |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 15 | |
reed | f037e0b | 2014-10-30 11:34:15 -0700 | [diff] [blame] | 16 | #if SK_SUPPORT_GPU |
| 17 | |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 18 | /////////////////////////////////////////////////////////////////////////////// |
| 19 | |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 20 | SkSurface_Gpu::SkSurface_Gpu(SkGpuDevice* device) |
| 21 | : INHERITED(device->width(), device->height(), &device->surfaceProps()) |
| 22 | , fDevice(SkRef(device)) { |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 23 | } |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 24 | |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 25 | SkSurface_Gpu::~SkSurface_Gpu() { |
kkinnunen | abcfab4 | 2015-02-22 22:53:44 -0800 | [diff] [blame^] | 26 | fDevice->unref(); |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 27 | } |
| 28 | |
| 29 | SkCanvas* SkSurface_Gpu::onNewCanvas() { |
reed | 4a8126e | 2014-09-22 07:29:03 -0700 | [diff] [blame] | 30 | SkCanvas::InitFlags flags = SkCanvas::kDefault_InitFlags; |
| 31 | // When we think this works... |
| 32 | // flags |= SkCanvas::kConservativeRasterClip_InitFlag; |
| 33 | |
| 34 | return SkNEW_ARGS(SkCanvas, (fDevice, &this->props(), flags)); |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 35 | } |
| 36 | |
reed@google.com | 2bd8b81 | 2013-11-01 13:46:54 +0000 | [diff] [blame] | 37 | SkSurface* SkSurface_Gpu::onNewSurface(const SkImageInfo& info) { |
commit-bot@chromium.org | b8d00db | 2013-06-26 19:18:23 +0000 | [diff] [blame] | 38 | GrRenderTarget* rt = fDevice->accessRenderTarget(); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 39 | int sampleCount = rt->numSamples(); |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 40 | // TODO: Make caller specify this (change virtual signature of onNewSurface). |
| 41 | static const Budgeted kBudgeted = kNo_Budgeted; |
| 42 | return SkSurface::NewRenderTarget(fDevice->context(), kBudgeted, info, sampleCount, |
| 43 | &this->props()); |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 44 | } |
| 45 | |
bsalomon | eaaaf0b | 2015-01-23 08:08:04 -0800 | [diff] [blame] | 46 | SkImage* SkSurface_Gpu::onNewImageSnapshot(Budgeted budgeted) { |
reed | 4af267b | 2014-11-21 08:46:37 -0800 | [diff] [blame] | 47 | const int sampleCount = fDevice->accessRenderTarget()->numSamples(); |
bsalomon | eaaaf0b | 2015-01-23 08:08:04 -0800 | [diff] [blame] | 48 | SkImage* image = SkNewImageFromBitmapTexture(fDevice->accessBitmap(false), sampleCount, |
| 49 | budgeted); |
reed | 4af267b | 2014-11-21 08:46:37 -0800 | [diff] [blame] | 50 | if (image) { |
| 51 | as_IB(image)->initWithProps(this->props()); |
| 52 | } |
| 53 | return image; |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 54 | } |
| 55 | |
| 56 | void SkSurface_Gpu::onDraw(SkCanvas* canvas, SkScalar x, SkScalar y, |
| 57 | const SkPaint* paint) { |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 58 | canvas->drawBitmap(fDevice->accessBitmap(false), x, y, paint); |
| 59 | } |
| 60 | |
kkinnunen | abcfab4 | 2015-02-22 22:53:44 -0800 | [diff] [blame^] | 61 | // Create a new render target and, if necessary, copy the contents of the old |
| 62 | // render target into it. Note that this flushes the SkGpuDevice but |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 63 | // doesn't force an OpenGL flush. |
commit-bot@chromium.org | c4c9870 | 2013-04-22 14:28:01 +0000 | [diff] [blame] | 64 | void SkSurface_Gpu::onCopyOnWrite(ContentChangeMode mode) { |
commit-bot@chromium.org | b8d00db | 2013-06-26 19:18:23 +0000 | [diff] [blame] | 65 | GrRenderTarget* rt = fDevice->accessRenderTarget(); |
bsalomon | eaaaf0b | 2015-01-23 08:08:04 -0800 | [diff] [blame] | 66 | // are we sharing our render target with the image? Note this call should never create a new |
| 67 | // image because onCopyOnWrite is only called when there is a cached image. |
| 68 | SkImage* image = this->getCachedImage(kNo_Budgeted); |
| 69 | SkASSERT(image); |
| 70 | if (rt->asTexture() == SkTextureImageGetTexture(image)) { |
kkinnunen | abcfab4 | 2015-02-22 22:53:44 -0800 | [diff] [blame^] | 71 | this->fDevice->replaceRenderTarget(SkSurface::kRetain_ContentChangeMode == mode); |
bsalomon | eaaaf0b | 2015-01-23 08:08:04 -0800 | [diff] [blame] | 72 | SkTextureImageApplyBudgetedDecision(image); |
commit-bot@chromium.org | 28361fa | 2014-03-28 16:08:05 +0000 | [diff] [blame] | 73 | } else if (kDiscard_ContentChangeMode == mode) { |
| 74 | this->SkSurface_Gpu::onDiscard(); |
robertphillips@google.com | 97b6b07 | 2012-10-31 14:48:39 +0000 | [diff] [blame] | 75 | } |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 76 | } |
| 77 | |
commit-bot@chromium.org | 28361fa | 2014-03-28 16:08:05 +0000 | [diff] [blame] | 78 | void SkSurface_Gpu::onDiscard() { |
| 79 | fDevice->accessRenderTarget()->discard(); |
| 80 | } |
| 81 | |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 82 | /////////////////////////////////////////////////////////////////////////////// |
| 83 | |
reed | 4a8126e | 2014-09-22 07:29:03 -0700 | [diff] [blame] | 84 | SkSurface* SkSurface::NewRenderTargetDirect(GrRenderTarget* target, const SkSurfaceProps* props) { |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 85 | SkAutoTUnref<SkGpuDevice> device(SkGpuDevice::Create(target, props)); |
| 86 | if (!device) { |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 87 | return NULL; |
| 88 | } |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 89 | return SkNEW_ARGS(SkSurface_Gpu, (device)); |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 90 | } |
| 91 | |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 92 | SkSurface* SkSurface::NewRenderTarget(GrContext* ctx, Budgeted budgeted, const SkImageInfo& info, |
| 93 | int sampleCount, const SkSurfaceProps* props) { |
| 94 | SkAutoTUnref<SkGpuDevice> device(SkGpuDevice::Create(ctx, budgeted, info, sampleCount, props, |
| 95 | SkGpuDevice::kNeedClear_Flag)); |
| 96 | if (!device) { |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 97 | return NULL; |
| 98 | } |
bsalomon | afe3005 | 2015-01-16 07:32:33 -0800 | [diff] [blame] | 99 | return SkNEW_ARGS(SkSurface_Gpu, (device)); |
reed@google.com | 5d4ba88 | 2012-07-31 15:45:27 +0000 | [diff] [blame] | 100 | } |
reed | f037e0b | 2014-10-30 11:34:15 -0700 | [diff] [blame] | 101 | |
| 102 | #endif |