blob: ab1d5c2546ed05029655ca8fb83c301b414654cf [file] [log] [blame]
Stan Iliev021693b2016-10-17 16:26:15 -04001/*
2 * Copyright (C) 2016 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include "LayerDrawable.h"
John Reck1bcacfd2017-11-03 10:12:19 -070018#include "GlLayer.h"
Greg Daniel45ec62b2017-01-04 14:27:00 -050019#include "VkLayer.h"
20
Greg Danielac2d2322017-07-12 11:30:15 -040021#include "GrBackendSurface.h"
Derek Sollenbergerf87da672016-11-02 11:34:27 -040022#include "SkColorFilter.h"
Greg Daniel45ec62b2017-01-04 14:27:00 -050023#include "SkSurface.h"
Stan Iliev021693b2016-10-17 16:26:15 -040024#include "gl/GrGLTypes.h"
25
26namespace android {
27namespace uirenderer {
28namespace skiapipeline {
29
30void LayerDrawable::onDraw(SkCanvas* canvas) {
Derek Sollenbergerf5a370e2017-06-15 13:50:08 -040031 Layer* layer = mLayerUpdater->backingLayer();
32 if (layer) {
33 DrawLayer(canvas->getGrContext(), canvas, layer);
34 }
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -050035}
36
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040037bool LayerDrawable::DrawLayer(GrContext* context, SkCanvas* canvas, Layer* layer,
38 const SkRect* dstRect) {
Stan Ilieve9d00122017-09-19 12:07:10 -040039 if (context == nullptr) {
40 SkDEBUGF(("Attempting to draw LayerDrawable into an unsupported surface"));
41 return false;
42 }
Stan Iliev021693b2016-10-17 16:26:15 -040043 // transform the matrix based on the layer
Stan Iliev944dbf22017-09-27 11:05:29 -040044 SkMatrix layerTransform;
45 layer->getTransform().copyTo(layerTransform);
Greg Daniel45ec62b2017-01-04 14:27:00 -050046 sk_sp<SkImage> layerImage;
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040047 const int layerWidth = layer->getWidth();
48 const int layerHeight = layer->getHeight();
Stan Iliev93ac4012018-09-10 14:48:38 -040049 const int bufferWidth = layer->getBufferWidth();
50 const int bufferHeight = layer->getBufferHeight();
51 if (bufferWidth <= 0 || bufferHeight <=0) {
52 return false;
53 }
Greg Daniel45ec62b2017-01-04 14:27:00 -050054 if (layer->getApi() == Layer::Api::OpenGL) {
55 GlLayer* glLayer = static_cast<GlLayer*>(layer);
56 GrGLTextureInfo externalTexture;
57 externalTexture.fTarget = glLayer->getRenderTarget();
58 externalTexture.fID = glLayer->getTextureId();
Derek Sollenberger551d08e2018-04-20 16:13:31 -040059 // The format may not be GL_RGBA8, but given the DeferredLayerUpdater and GLConsumer don't
60 // expose that info we use it as our default. Further, given that we only use this texture
61 // as a source this will not impact how Skia uses the texture. The only potential affect
62 // this is anticipated to have is that for some format types if we are not bound as an OES
63 // texture we may get invalid results for SKP capture if we read back the texture.
64 externalTexture.fFormat = GL_RGBA8;
Stan Iliev93ac4012018-09-10 14:48:38 -040065 GrBackendTexture backendTexture(bufferWidth, bufferHeight, GrMipMapped::kNo, externalTexture);
Greg Danielac2d2322017-07-12 11:30:15 -040066 layerImage = SkImage::MakeFromTexture(context, backendTexture, kTopLeft_GrSurfaceOrigin,
John Reck1bcacfd2017-11-03 10:12:19 -070067 kPremul_SkAlphaType, nullptr);
Greg Daniel45ec62b2017-01-04 14:27:00 -050068 } else {
69 SkASSERT(layer->getApi() == Layer::Api::Vulkan);
70 VkLayer* vkLayer = static_cast<VkLayer*>(layer);
71 canvas->clear(SK_ColorGREEN);
72 layerImage = vkLayer->getImage();
73 }
74
Stan Iliev021693b2016-10-17 16:26:15 -040075 if (layerImage) {
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040076 SkMatrix textureMatrixInv;
77 layer->getTexTransform().copyTo(textureMatrixInv);
John Reck1bcacfd2017-11-03 10:12:19 -070078 // TODO: after skia bug https://bugs.chromium.org/p/skia/issues/detail?id=7075 is fixed
Stan Iliev944dbf22017-09-27 11:05:29 -040079 // use bottom left origin and remove flipV and invert transformations.
80 SkMatrix flipV;
81 flipV.setAll(1, 0, 0, 0, -1, 1, 0, 0, 1);
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040082 textureMatrixInv.preConcat(flipV);
83 textureMatrixInv.preScale(1.0f / layerWidth, 1.0f / layerHeight);
Stan Iliev93ac4012018-09-10 14:48:38 -040084 textureMatrixInv.postScale(bufferWidth, bufferHeight);
Leon Scroggins III1a12ab22018-03-26 15:00:49 -040085 SkMatrix textureMatrix;
86 if (!textureMatrixInv.invert(&textureMatrix)) {
87 textureMatrix = textureMatrixInv;
Stan Iliev944dbf22017-09-27 11:05:29 -040088 }
89
Stan Ilievff129ae2018-07-12 16:53:59 -040090 SkMatrix matrix;
91 if (dstRect) {
92 // Destination rectangle is set only when we are trying to read back the content
93 // of the layer. In this case we don't want to apply layer transform.
94 matrix = textureMatrix;
95 } else {
96 matrix = SkMatrix::Concat(layerTransform, textureMatrix);
97 }
Stan Iliev944dbf22017-09-27 11:05:29 -040098
Stan Iliev021693b2016-10-17 16:26:15 -040099 SkPaint paint;
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -0500100 paint.setAlpha(layer->getAlpha());
101 paint.setBlendMode(layer->getMode());
Derek Sollenberger551d08e2018-04-20 16:13:31 -0400102 paint.setColorFilter(layer->getColorSpaceWithFilter());
Stan Ilieva73b0be2017-10-06 10:16:58 -0400103
104 const bool nonIdentityMatrix = !matrix.isIdentity();
105 if (nonIdentityMatrix) {
106 canvas->save();
107 canvas->concat(matrix);
108 }
Stan Iliev402fff12018-10-25 16:28:27 -0400109 const SkMatrix& totalMatrix = canvas->getTotalMatrix();
Leon Scroggins III1a12ab22018-03-26 15:00:49 -0400110 if (dstRect) {
111 SkMatrix matrixInv;
112 if (!matrix.invert(&matrixInv)) {
113 matrixInv = matrix;
114 }
115 SkRect srcRect = SkRect::MakeIWH(layerWidth, layerHeight);
116 matrixInv.mapRect(&srcRect);
117 SkRect skiaDestRect = *dstRect;
118 matrixInv.mapRect(&skiaDestRect);
Stan Iliev402fff12018-10-25 16:28:27 -0400119 // If (matrix is identity or an integer translation) and (src/dst buffers size match),
120 // then use nearest neighbor, otherwise use bilerp sampling.
121 // Integer translation is defined as when src rect and dst rect align fractionally.
122 // Skia TextureOp has the above logic build-in, but not NonAAFillRectOp. TextureOp works
123 // only for SrcOver blending and without color filter (readback uses Src blending).
124 bool isIntegerTranslate = totalMatrix.isTranslate()
125 && SkScalarFraction(skiaDestRect.fLeft + totalMatrix[SkMatrix::kMTransX])
126 == SkScalarFraction(srcRect.fLeft)
127 && SkScalarFraction(skiaDestRect.fTop + totalMatrix[SkMatrix::kMTransY])
128 == SkScalarFraction(srcRect.fTop);
129 if (layer->getForceFilter() || !isIntegerTranslate) {
130 paint.setFilterQuality(kLow_SkFilterQuality);
131 }
Leon Scroggins III1a12ab22018-03-26 15:00:49 -0400132 canvas->drawImageRect(layerImage.get(), srcRect, skiaDestRect, &paint,
133 SkCanvas::kFast_SrcRectConstraint);
134 } else {
Stan Iliev402fff12018-10-25 16:28:27 -0400135 bool isIntegerTranslate = totalMatrix.isTranslate()
136 && SkScalarIsInt(totalMatrix[SkMatrix::kMTransX])
137 && SkScalarIsInt(totalMatrix[SkMatrix::kMTransY]);
138 if (layer->getForceFilter() || !isIntegerTranslate) {
139 paint.setFilterQuality(kLow_SkFilterQuality);
140 }
Leon Scroggins III1a12ab22018-03-26 15:00:49 -0400141 canvas->drawImage(layerImage.get(), 0, 0, &paint);
142 }
Stan Ilieva73b0be2017-10-06 10:16:58 -0400143 // restore the original matrix
144 if (nonIdentityMatrix) {
145 canvas->restore();
146 }
Stan Iliev021693b2016-10-17 16:26:15 -0400147 }
Derek Sollenbergerc4fbada2016-11-07 16:05:41 -0500148
149 return layerImage;
Stan Iliev021693b2016-10-17 16:26:15 -0400150}
151
John Reck1bcacfd2017-11-03 10:12:19 -0700152}; // namespace skiapipeline
153}; // namespace uirenderer
154}; // namespace android