blob: 1f7092147686c1f09d635abe2d71f48353cc4179 [file] [log] [blame]
Romain Guy4aa90572010-09-26 18:40:37 -07001/*
2 * Copyright (C) 2010 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#define LOG_TAG "OpenGLRenderer"
18
Romain Guyd5a85fb2012-03-13 11:18:20 -070019#include <SkCamera.h>
Romain Guyc46d07a2013-03-15 19:06:39 -070020#include <SkCanvas.h>
Chet Haase9c1e23b2011-03-24 10:51:31 -070021
Romain Guy65549432012-03-26 16:45:05 -070022#include <private/hwui/DrawGlInfo.h>
23
John Reck113e0822014-03-18 09:22:59 -070024#include "Caches.h"
Chris Craikc3566d02013-02-04 16:16:33 -080025#include "DeferredDisplayList.h"
Chet Haase9c1e23b2011-03-24 10:51:31 -070026#include "DisplayListLogBuffer.h"
Chris Craik2af46352012-11-26 18:30:17 -080027#include "DisplayListOp.h"
Romain Guy4aa90572010-09-26 18:40:37 -070028#include "DisplayListRenderer.h"
John Reck113e0822014-03-18 09:22:59 -070029#include "RenderNode.h"
Romain Guy13631f32012-01-30 17:41:55 -080030
Romain Guy4aa90572010-09-26 18:40:37 -070031namespace android {
32namespace uirenderer {
33
Chris Craikcce47eb2014-07-16 15:12:15 -070034DisplayListRenderer::DisplayListRenderer()
35 : mCaches(Caches::getInstance())
Chris Craik8afd0f22014-08-21 17:41:57 -070036 , mDisplayListData(NULL)
Chris Craikcce47eb2014-07-16 15:12:15 -070037 , mTranslateX(0.0f)
38 , mTranslateY(0.0f)
Chris Craik8afd0f22014-08-21 17:41:57 -070039 , mDeferredBarrierType(kBarrier_None)
Chris Craikcce47eb2014-07-16 15:12:15 -070040 , mHighContrastText(false)
41 , mRestoreSaveCount(-1) {
Romain Guy4aa90572010-09-26 18:40:37 -070042}
43
44DisplayListRenderer::~DisplayListRenderer() {
John Reck44fd8d22014-02-26 11:00:11 -080045 LOG_ALWAYS_FATAL_IF(mDisplayListData,
46 "Destroyed a DisplayListRenderer during a record!");
Romain Guy4aa90572010-09-26 18:40:37 -070047}
48
49///////////////////////////////////////////////////////////////////////////////
50// Operations
51///////////////////////////////////////////////////////////////////////////////
52
John Reck44fd8d22014-02-26 11:00:11 -080053DisplayListData* DisplayListRenderer::finishRecording() {
John Reck44fd8d22014-02-26 11:00:11 -080054 mPaintMap.clear();
55 mRegionMap.clear();
56 mPathMap.clear();
57 DisplayListData* data = mDisplayListData;
58 mDisplayListData = 0;
59 return data;
Chet Haase5977baa2011-01-05 18:01:22 -080060}
61
Romain Guy7c25aab2012-10-18 15:05:02 -070062status_t DisplayListRenderer::prepareDirty(float left, float top,
Romain Guy7d7b5492011-01-24 16:33:45 -080063 float right, float bottom, bool opaque) {
John Reck44fd8d22014-02-26 11:00:11 -080064
65 LOG_ALWAYS_FATAL_IF(mDisplayListData,
66 "prepareDirty called a second time during a recording!");
67 mDisplayListData = new DisplayListData();
68
Chris Craik69e5adf2014-08-14 13:34:01 -070069 initializeSaveStack(0, 0, getWidth(), getHeight(), Vector3());
Romain Guy45e4c3d2012-09-11 17:17:07 -070070
Chris Craik8afd0f22014-08-21 17:41:57 -070071 mDeferredBarrierType = kBarrier_InOrder;
Romain Guy45e4c3d2012-09-11 17:17:07 -070072 mDirtyClip = opaque;
Romain Guy27454a42011-01-23 12:01:41 -080073 mRestoreSaveCount = -1;
Romain Guy45e4c3d2012-09-11 17:17:07 -070074
Chet Haase44b2fe32012-06-06 19:03:58 -070075 return DrawGlInfo::kStatusDone; // No invalidate needed at record-time
Romain Guy27454a42011-01-23 12:01:41 -080076}
77
78void DisplayListRenderer::finish() {
Chris Craik8afd0f22014-08-21 17:41:57 -070079 flushRestoreToCount();
80 flushTranslate();
Romain Guyb051e892010-09-28 19:09:36 -070081}
82
Chet Haasedaf98e92011-01-10 14:10:36 -080083void DisplayListRenderer::interrupt() {
Chet Haasedaf98e92011-01-10 14:10:36 -080084}
Romain Guy2b1847e2011-01-26 13:43:01 -080085
Chet Haasedaf98e92011-01-10 14:10:36 -080086void DisplayListRenderer::resume() {
Romain Guy4aa90572010-09-26 18:40:37 -070087}
88
Romain Guy65549432012-03-26 16:45:05 -070089status_t DisplayListRenderer::callDrawGLFunction(Functor *functor, Rect& dirty) {
Romain Guycabfcc12011-03-07 18:06:46 -080090 // Ignore dirty during recording, it matters only when we replay
Chris Craik2af46352012-11-26 18:30:17 -080091 addDrawOp(new (alloc()) DrawFunctorOp(functor));
John Reck09d5cdd2014-07-24 10:36:08 -070092 mDisplayListData->functors.add(functor);
Romain Guy65549432012-03-26 16:45:05 -070093 return DrawGlInfo::kStatusDone; // No invalidate needed at record-time
Chet Haasedaf98e92011-01-10 14:10:36 -080094}
95
Romain Guy4aa90572010-09-26 18:40:37 -070096int DisplayListRenderer::save(int flags) {
Chris Craik2af46352012-11-26 18:30:17 -080097 addStateOp(new (alloc()) SaveOp(flags));
Chris Craik14e51302013-12-30 15:32:54 -080098 return StatefulBaseRenderer::save(flags);
Romain Guy4aa90572010-09-26 18:40:37 -070099}
100
101void DisplayListRenderer::restore() {
Romain Guy04c9d8c2011-08-25 14:01:48 -0700102 if (mRestoreSaveCount < 0) {
Romain Guy33f6beb2012-02-16 19:24:51 -0800103 restoreToCount(getSaveCount() - 1);
104 return;
Romain Guy04c9d8c2011-08-25 14:01:48 -0700105 }
Romain Guy33f6beb2012-02-16 19:24:51 -0800106
107 mRestoreSaveCount--;
Chris Craik8afd0f22014-08-21 17:41:57 -0700108 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800109 StatefulBaseRenderer::restore();
Romain Guy4aa90572010-09-26 18:40:37 -0700110}
111
112void DisplayListRenderer::restoreToCount(int saveCount) {
Romain Guy27454a42011-01-23 12:01:41 -0800113 mRestoreSaveCount = saveCount;
Chris Craik8afd0f22014-08-21 17:41:57 -0700114 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800115 StatefulBaseRenderer::restoreToCount(saveCount);
Romain Guy4aa90572010-09-26 18:40:37 -0700116}
117
118int DisplayListRenderer::saveLayer(float left, float top, float right, float bottom,
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500119 const SkPaint* paint, int flags) {
Chris Craik4ace7302014-09-14 15:49:54 -0700120 // force matrix/clip isolation for layer
121 flags |= SkCanvas::kClip_SaveFlag | SkCanvas::kMatrix_SaveFlag;
122
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500123 paint = refPaint(paint);
124 addStateOp(new (alloc()) SaveLayerOp(left, top, right, bottom, paint, flags));
Chris Craik14e51302013-12-30 15:32:54 -0800125 return StatefulBaseRenderer::save(flags);
Romain Guy5b3b3522010-10-27 18:57:51 -0700126}
127
Chris Craikb4589422013-12-26 15:13:13 -0800128void DisplayListRenderer::translate(float dx, float dy, float dz) {
129 // ignore dz, not used at defer time
Chris Craik8afd0f22014-08-21 17:41:57 -0700130 mHasDeferredTranslate = true;
Romain Guy33f6beb2012-02-16 19:24:51 -0800131 mTranslateX += dx;
132 mTranslateY += dy;
Chris Craik8afd0f22014-08-21 17:41:57 -0700133 flushRestoreToCount();
Chris Craik14e51302013-12-30 15:32:54 -0800134 StatefulBaseRenderer::translate(dx, dy, dz);
Romain Guy4aa90572010-09-26 18:40:37 -0700135}
136
137void DisplayListRenderer::rotate(float degrees) {
Chris Craik2af46352012-11-26 18:30:17 -0800138 addStateOp(new (alloc()) RotateOp(degrees));
Chris Craik14e51302013-12-30 15:32:54 -0800139 StatefulBaseRenderer::rotate(degrees);
Romain Guy4aa90572010-09-26 18:40:37 -0700140}
141
142void DisplayListRenderer::scale(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800143 addStateOp(new (alloc()) ScaleOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800144 StatefulBaseRenderer::scale(sx, sy);
Romain Guy4aa90572010-09-26 18:40:37 -0700145}
146
Romain Guy807daf72011-01-18 11:19:19 -0800147void DisplayListRenderer::skew(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800148 addStateOp(new (alloc()) SkewOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800149 StatefulBaseRenderer::skew(sx, sy);
Romain Guy807daf72011-01-18 11:19:19 -0800150}
151
Derek Sollenberger13908822013-12-10 12:28:58 -0500152void DisplayListRenderer::setMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800153 addStateOp(new (alloc()) SetMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800154 StatefulBaseRenderer::setMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700155}
156
Derek Sollenberger13908822013-12-10 12:28:58 -0500157void DisplayListRenderer::concatMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800158 addStateOp(new (alloc()) ConcatMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800159 StatefulBaseRenderer::concatMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700160}
161
162bool DisplayListRenderer::clipRect(float left, float top, float right, float bottom,
163 SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800164 addStateOp(new (alloc()) ClipRectOp(left, top, right, bottom, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800165 return StatefulBaseRenderer::clipRect(left, top, right, bottom, op);
Romain Guy4aa90572010-09-26 18:40:37 -0700166}
167
Chris Craikd218a922014-01-02 17:13:34 -0800168bool DisplayListRenderer::clipPath(const SkPath* path, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800169 path = refPath(path);
170 addStateOp(new (alloc()) ClipPathOp(path, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800171 return StatefulBaseRenderer::clipPath(path, op);
Romain Guy735738c2012-12-03 12:34:51 -0800172}
173
Chris Craikd218a922014-01-02 17:13:34 -0800174bool DisplayListRenderer::clipRegion(const SkRegion* region, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800175 region = refRegion(region);
176 addStateOp(new (alloc()) ClipRegionOp(region, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800177 return StatefulBaseRenderer::clipRegion(region, op);
Romain Guy735738c2012-12-03 12:34:51 -0800178}
179
Chris Craika7090e02014-06-20 16:01:00 -0700180status_t DisplayListRenderer::drawRenderNode(RenderNode* renderNode, Rect& dirty, int32_t flags) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700181 LOG_ALWAYS_FATAL_IF(!renderNode, "missing rendernode");
182
Romain Guycabfcc12011-03-07 18:06:46 -0800183 // dirty is an out parameter and should not be recorded,
184 // it matters only when replaying the display list
Chris Craikb3cca872014-08-08 18:42:51 -0700185 DrawRenderNodeOp* op = new (alloc()) DrawRenderNodeOp(renderNode, flags, *currentTransform());
Chris Craik8afd0f22014-08-21 17:41:57 -0700186 addRenderNodeOp(op);
Chris Craik2af46352012-11-26 18:30:17 -0800187
Romain Guy65549432012-03-26 16:45:05 -0700188 return DrawGlInfo::kStatusDone;
Romain Guy0fe478e2010-11-08 12:08:41 -0800189}
190
Chris Craika08f95c2013-03-15 17:24:33 -0700191status_t DisplayListRenderer::drawLayer(Layer* layer, float x, float y) {
Romain Guyce4a7df2013-03-28 11:32:33 -0700192 layer = refLayer(layer);
Chris Craika08f95c2013-03-15 17:24:33 -0700193 addDrawOp(new (alloc()) DrawLayerOp(layer, x, y));
Chet Haase48659092012-05-31 15:21:51 -0700194 return DrawGlInfo::kStatusDone;
Romain Guy6c319ca2011-01-11 14:29:25 -0800195}
196
Chris Craik79647502014-08-06 13:42:24 -0700197status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800198 bitmap = refBitmap(bitmap);
199 paint = refPaint(paint);
200
Chris Craik79647502014-08-06 13:42:24 -0700201 addDrawOp(new (alloc()) DrawBitmapOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700202 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700203}
204
Chris Craikd218a922014-01-02 17:13:34 -0800205status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, float srcLeft, float srcTop,
Romain Guy4aa90572010-09-26 18:40:37 -0700206 float srcRight, float srcBottom, float dstLeft, float dstTop,
Chris Craikd218a922014-01-02 17:13:34 -0800207 float dstRight, float dstBottom, const SkPaint* paint) {
Chris Craik79647502014-08-06 13:42:24 -0700208 if (srcLeft == 0 && srcTop == 0
209 && srcRight == bitmap->width() && srcBottom == bitmap->height()
210 && (srcBottom - srcTop == dstBottom - dstTop)
211 && (srcRight - srcLeft == dstRight - dstLeft)) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800212 // transform simple rect to rect drawing case into position bitmap ops, since they merge
Chris Craik79647502014-08-06 13:42:24 -0700213 save(SkCanvas::kMatrix_SaveFlag);
214 translate(dstLeft, dstTop);
215 drawBitmap(bitmap, paint);
216 restore();
217 } else {
218 bitmap = refBitmap(bitmap);
219 paint = refPaint(paint);
Chris Craik527a3aa2013-03-04 10:19:31 -0800220
Chris Craik79647502014-08-06 13:42:24 -0700221 addDrawOp(new (alloc()) DrawBitmapRectOp(bitmap,
222 srcLeft, srcTop, srcRight, srcBottom,
223 dstLeft, dstTop, dstRight, dstBottom, paint));
224 }
Chet Haase48659092012-05-31 15:21:51 -0700225 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700226}
227
Chris Craik79647502014-08-06 13:42:24 -0700228status_t DisplayListRenderer::drawBitmapData(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800229 bitmap = refBitmapData(bitmap);
230 paint = refPaint(paint);
231
Chris Craik79647502014-08-06 13:42:24 -0700232 addDrawOp(new (alloc()) DrawBitmapDataOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700233 return DrawGlInfo::kStatusDone;
Romain Guye651cc62012-05-14 19:44:40 -0700234}
235
Chris Craikd218a922014-01-02 17:13:34 -0800236status_t DisplayListRenderer::drawBitmapMesh(const SkBitmap* bitmap, int meshWidth, int meshHeight,
237 const float* vertices, const int* colors, const SkPaint* paint) {
Chris Craik0664fef2014-04-11 13:40:05 -0700238 int vertexCount = (meshWidth + 1) * (meshHeight + 1);
Chris Craik2af46352012-11-26 18:30:17 -0800239 bitmap = refBitmap(bitmap);
Chris Craik0664fef2014-04-11 13:40:05 -0700240 vertices = refBuffer<float>(vertices, vertexCount * 2); // 2 floats per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800241 paint = refPaint(paint);
Chris Craik0664fef2014-04-11 13:40:05 -0700242 colors = refBuffer<int>(colors, vertexCount); // 1 color per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800243
244 addDrawOp(new (alloc()) DrawBitmapMeshOp(bitmap, meshWidth, meshHeight,
245 vertices, colors, paint));
Chet Haase48659092012-05-31 15:21:51 -0700246 return DrawGlInfo::kStatusDone;
Romain Guy5a7b4662011-01-20 19:09:30 -0800247}
248
Chris Craikd218a922014-01-02 17:13:34 -0800249status_t DisplayListRenderer::drawPatch(const SkBitmap* bitmap, const Res_png_9patch* patch,
250 float left, float top, float right, float bottom, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800251 bitmap = refBitmap(bitmap);
Romain Guye3b0a012013-06-26 15:45:41 -0700252 patch = refPatch(patch);
Romain Guy16ea8d32013-06-21 11:35:52 -0700253 paint = refPaint(paint);
Chris Craik2af46352012-11-26 18:30:17 -0800254
Romain Guy03c00b52013-06-20 18:30:28 -0700255 addDrawOp(new (alloc()) DrawPatchOp(bitmap, patch, left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700256 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700257}
258
Chet Haase48659092012-05-31 15:21:51 -0700259status_t DisplayListRenderer::drawColor(int color, SkXfermode::Mode mode) {
Chris Craik2af46352012-11-26 18:30:17 -0800260 addDrawOp(new (alloc()) DrawColorOp(color, mode));
Chet Haase48659092012-05-31 15:21:51 -0700261 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700262}
263
Chet Haase48659092012-05-31 15:21:51 -0700264status_t DisplayListRenderer::drawRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800265 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800266 paint = refPaint(paint);
267 addDrawOp(new (alloc()) DrawRectOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700268 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700269}
270
Chet Haase48659092012-05-31 15:21:51 -0700271status_t DisplayListRenderer::drawRoundRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800272 float rx, float ry, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800273 paint = refPaint(paint);
274 addDrawOp(new (alloc()) DrawRoundRectOp(left, top, right, bottom, rx, ry, paint));
Chet Haase48659092012-05-31 15:21:51 -0700275 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800276}
277
Jorim Jaggi072707d2014-09-15 17:20:08 +0200278status_t DisplayListRenderer::drawRoundRect(
279 CanvasPropertyPrimitive* left, CanvasPropertyPrimitive* top,
280 CanvasPropertyPrimitive* right, CanvasPropertyPrimitive* bottom,
281 CanvasPropertyPrimitive* rx, CanvasPropertyPrimitive* ry,
282 CanvasPropertyPaint* paint) {
283 mDisplayListData->refProperty(left);
284 mDisplayListData->refProperty(top);
285 mDisplayListData->refProperty(right);
286 mDisplayListData->refProperty(bottom);
287 mDisplayListData->refProperty(rx);
288 mDisplayListData->refProperty(ry);
289 mDisplayListData->refProperty(paint);
290 addDrawOp(new (alloc()) DrawRoundRectPropsOp(&left->value, &top->value,
291 &right->value, &bottom->value, &rx->value, &ry->value, &paint->value));
292 return DrawGlInfo::kStatusDone;
293}
294
Chris Craikd218a922014-01-02 17:13:34 -0800295status_t DisplayListRenderer::drawCircle(float x, float y, float radius, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800296 paint = refPaint(paint);
297 addDrawOp(new (alloc()) DrawCircleOp(x, y, radius, paint));
Chet Haase48659092012-05-31 15:21:51 -0700298 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800299}
300
John Reck52244ff2014-05-01 21:27:37 -0700301status_t DisplayListRenderer::drawCircle(CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y,
302 CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) {
303 mDisplayListData->refProperty(x);
304 mDisplayListData->refProperty(y);
305 mDisplayListData->refProperty(radius);
306 mDisplayListData->refProperty(paint);
307 addDrawOp(new (alloc()) DrawCirclePropsOp(&x->value, &y->value,
308 &radius->value, &paint->value));
309 return DrawGlInfo::kStatusDone;
310}
311
Chet Haase48659092012-05-31 15:21:51 -0700312status_t DisplayListRenderer::drawOval(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800313 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800314 paint = refPaint(paint);
315 addDrawOp(new (alloc()) DrawOvalOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700316 return DrawGlInfo::kStatusDone;
Romain Guyc1cd9ba32011-01-23 14:18:41 -0800317}
318
Chet Haase48659092012-05-31 15:21:51 -0700319status_t DisplayListRenderer::drawArc(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800320 float startAngle, float sweepAngle, bool useCenter, const SkPaint* paint) {
Chris Craik544e5242014-07-11 10:58:10 -0700321 if (fabs(sweepAngle) >= 360.0f) {
Chris Craik6ac174b2014-06-17 13:47:05 -0700322 return drawOval(left, top, right, bottom, paint);
323 }
324
Chris Craik2af46352012-11-26 18:30:17 -0800325 paint = refPaint(paint);
326 addDrawOp(new (alloc()) DrawArcOp(left, top, right, bottom,
327 startAngle, sweepAngle, useCenter, paint));
Chet Haase48659092012-05-31 15:21:51 -0700328 return DrawGlInfo::kStatusDone;
Romain Guy8b2f5262011-01-23 16:15:02 -0800329}
330
Chris Craikd218a922014-01-02 17:13:34 -0800331status_t DisplayListRenderer::drawPath(const SkPath* path, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800332 path = refPath(path);
333 paint = refPaint(paint);
Romain Guy33f6beb2012-02-16 19:24:51 -0800334
Chris Craik2af46352012-11-26 18:30:17 -0800335 addDrawOp(new (alloc()) DrawPathOp(path, paint));
Chet Haase48659092012-05-31 15:21:51 -0700336 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700337}
338
Chris Craikd218a922014-01-02 17:13:34 -0800339status_t DisplayListRenderer::drawLines(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800340 points = refBuffer<float>(points, count);
341 paint = refPaint(paint);
342
343 addDrawOp(new (alloc()) DrawLinesOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700344 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700345}
346
Chris Craikd218a922014-01-02 17:13:34 -0800347status_t DisplayListRenderer::drawPoints(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800348 points = refBuffer<float>(points, count);
349 paint = refPaint(paint);
350
351 addDrawOp(new (alloc()) DrawPointsOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700352 return DrawGlInfo::kStatusDone;
Romain Guyed6fcb02011-03-21 13:11:28 -0700353}
354
Chet Haase48659092012-05-31 15:21:51 -0700355status_t DisplayListRenderer::drawTextOnPath(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800356 const SkPath* path, float hOffset, float vOffset, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700357 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800358
Chris Craik2af46352012-11-26 18:30:17 -0800359 text = refText(text, bytesCount);
360 path = refPath(path);
361 paint = refPaint(paint);
362
363 DrawOp* op = new (alloc()) DrawTextOnPathOp(text, bytesCount, count, path,
364 hOffset, vOffset, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800365 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700366 return DrawGlInfo::kStatusDone;
Romain Guy325740f2012-02-24 16:48:34 -0800367}
368
Chet Haase48659092012-05-31 15:21:51 -0700369status_t DisplayListRenderer::drawPosText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800370 const float* positions, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700371 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800372
Chris Craik2af46352012-11-26 18:30:17 -0800373 text = refText(text, bytesCount);
374 positions = refBuffer<float>(positions, count * 2);
375 paint = refPaint(paint);
376
377 DrawOp* op = new (alloc()) DrawPosTextOp(text, bytesCount, count, positions, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800378 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700379 return DrawGlInfo::kStatusDone;
Romain Guyeb9a5362012-01-17 17:39:26 -0800380}
381
Chris Craikcce47eb2014-07-16 15:12:15 -0700382static void simplifyPaint(int color, SkPaint* paint) {
383 paint->setColor(color);
384 paint->setShader(NULL);
385 paint->setColorFilter(NULL);
386 paint->setLooper(NULL);
387 paint->setStrokeWidth(4 + 0.04 * paint->getTextSize());
388 paint->setStrokeJoin(SkPaint::kRound_Join);
389 paint->setLooper(NULL);
390}
391
Romain Guyc2525952012-07-27 16:41:22 -0700392status_t DisplayListRenderer::drawText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800393 float x, float y, const float* positions, const SkPaint* paint,
Chris Craik41541822013-05-03 16:35:54 -0700394 float totalAdvance, const Rect& bounds, DrawOpMode drawOpMode) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800395
Chris Craik947eabf2014-08-19 10:21:12 -0700396 if (!text || count <= 0 || paintWillNotDrawText(*paint)) return DrawGlInfo::kStatusDone;
Raph Levien996e57c2012-07-23 15:22:52 -0700397
Chris Craik2af46352012-11-26 18:30:17 -0800398 text = refText(text, bytesCount);
399 positions = refBuffer<float>(positions, count * 2);
Raph Levien996e57c2012-07-23 15:22:52 -0700400
Chris Craikcce47eb2014-07-16 15:12:15 -0700401 if (CC_UNLIKELY(mHighContrastText)) {
402 // high contrast draw path
403 int color = paint->getColor();
404 int channelSum = SkColorGetR(color) + SkColorGetG(color) + SkColorGetB(color);
405 bool darken = channelSum < (128 * 3);
406
407 // outline
408 SkPaint* outlinePaint = copyPaint(paint);
409 simplifyPaint(darken ? SK_ColorWHITE : SK_ColorBLACK, outlinePaint);
410 outlinePaint->setStyle(SkPaint::kStrokeAndFill_Style);
411 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
412 x, y, positions, outlinePaint, totalAdvance, bounds)); // bounds?
413
414 // inner
415 SkPaint* innerPaint = copyPaint(paint);
416 simplifyPaint(darken ? SK_ColorBLACK : SK_ColorWHITE, innerPaint);
417 innerPaint->setStyle(SkPaint::kFill_Style);
418 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
419 x, y, positions, innerPaint, totalAdvance, bounds));
420 } else {
421 // standard draw path
422 paint = refPaint(paint);
423
424 DrawOp* op = new (alloc()) DrawTextOp(text, bytesCount, count,
425 x, y, positions, paint, totalAdvance, bounds);
426 addDrawOp(op);
427 }
Raph Levien996e57c2012-07-23 15:22:52 -0700428 return DrawGlInfo::kStatusDone;
429}
430
Chris Craikd218a922014-01-02 17:13:34 -0800431status_t DisplayListRenderer::drawRects(const float* rects, int count, const SkPaint* paint) {
Romain Guy672433d2013-01-04 19:05:13 -0800432 if (count <= 0) return DrawGlInfo::kStatusDone;
433
Chris Craik2af46352012-11-26 18:30:17 -0800434 rects = refBuffer<float>(rects, count);
435 paint = refPaint(paint);
436 addDrawOp(new (alloc()) DrawRectsOp(rects, count, paint));
Romain Guy672433d2013-01-04 19:05:13 -0800437 return DrawGlInfo::kStatusDone;
438}
439
Romain Guy5ff9df62012-01-23 17:09:05 -0800440void DisplayListRenderer::resetPaintFilter() {
Chris Craik2af46352012-11-26 18:30:17 -0800441 addStateOp(new (alloc()) ResetPaintFilterOp());
Romain Guy5ff9df62012-01-23 17:09:05 -0800442}
443
444void DisplayListRenderer::setupPaintFilter(int clearBits, int setBits) {
Chris Craik2af46352012-11-26 18:30:17 -0800445 addStateOp(new (alloc()) SetupPaintFilterOp(clearBits, setBits));
446}
447
Chris Craik8afd0f22014-08-21 17:41:57 -0700448void DisplayListRenderer::insertReorderBarrier(bool enableReorder) {
449 flushRestoreToCount();
450 flushTranslate();
451 mDeferredBarrierType = enableReorder ? kBarrier_OutOfOrder : kBarrier_InOrder;
452}
453
454void DisplayListRenderer::flushRestoreToCount() {
Chris Craik2af46352012-11-26 18:30:17 -0800455 if (mRestoreSaveCount >= 0) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700456 addOpAndUpdateChunk(new (alloc()) RestoreToCountOp(mRestoreSaveCount));
Chris Craik2af46352012-11-26 18:30:17 -0800457 mRestoreSaveCount = -1;
458 }
459}
460
Chris Craik8afd0f22014-08-21 17:41:57 -0700461void DisplayListRenderer::flushTranslate() {
462 if (mHasDeferredTranslate) {
Chris Craik2af46352012-11-26 18:30:17 -0800463 if (mTranslateX != 0.0f || mTranslateY != 0.0f) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700464 addOpAndUpdateChunk(new (alloc()) TranslateOp(mTranslateX, mTranslateY));
Chris Craik2af46352012-11-26 18:30:17 -0800465 mTranslateX = mTranslateY = 0.0f;
466 }
Chris Craik8afd0f22014-08-21 17:41:57 -0700467 mHasDeferredTranslate = false;
Chris Craik2af46352012-11-26 18:30:17 -0800468 }
469}
470
Chris Craik8afd0f22014-08-21 17:41:57 -0700471size_t DisplayListRenderer::addOpAndUpdateChunk(DisplayListOp* op) {
472 int insertIndex = mDisplayListData->displayListOps.add(op);
473 if (mDeferredBarrierType != kBarrier_None) {
474 // op is first in new chunk
475 mDisplayListData->chunks.push();
476 DisplayListData::Chunk& newChunk = mDisplayListData->chunks.editTop();
477 newChunk.beginOpIndex = insertIndex;
478 newChunk.endOpIndex = insertIndex + 1;
479 newChunk.reorderChildren = (mDeferredBarrierType == kBarrier_OutOfOrder);
480
481 int nextChildIndex = mDisplayListData->children().size();
482 newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex;
483 mDeferredBarrierType = kBarrier_None;
484 } else {
485 // standard case - append to existing chunk
486 mDisplayListData->chunks.editTop().endOpIndex = insertIndex + 1;
487 }
488 return insertIndex;
Chris Craik2af46352012-11-26 18:30:17 -0800489}
490
Chris Craik8afd0f22014-08-21 17:41:57 -0700491size_t DisplayListRenderer::flushAndAddOp(DisplayListOp* op) {
492 flushRestoreToCount();
493 flushTranslate();
494 return addOpAndUpdateChunk(op);
495}
496
497size_t DisplayListRenderer::addStateOp(StateOp* op) {
498 return flushAndAddOp(op);
499}
500
501size_t DisplayListRenderer::addDrawOp(DrawOp* op) {
Chris Craik2af46352012-11-26 18:30:17 -0800502 Rect localBounds;
John Reck3b202512014-06-23 13:13:08 -0700503 if (op->getLocalBounds(localBounds)) {
Chris Craikf0a59072013-11-19 18:00:46 -0800504 bool rejected = quickRejectConservative(localBounds.left, localBounds.top,
Chris Craik2af46352012-11-26 18:30:17 -0800505 localBounds.right, localBounds.bottom);
506 op->setQuickRejected(rejected);
507 }
Chris Craikc1c5f082013-09-11 16:23:37 -0700508
John Reck44fd8d22014-02-26 11:00:11 -0800509 mDisplayListData->hasDrawOps = true;
Chris Craik8afd0f22014-08-21 17:41:57 -0700510 return flushAndAddOp(op);
511}
512
513size_t DisplayListRenderer::addRenderNodeOp(DrawRenderNodeOp* op) {
514 int opIndex = addDrawOp(op);
515 int childIndex = mDisplayListData->addChild(op);
516
517 // update the chunk's child indices
518 DisplayListData::Chunk& chunk = mDisplayListData->chunks.editTop();
519 chunk.endChildIndex = childIndex + 1;
520
521 if (op->renderNode()->stagingProperties().isProjectionReceiver()) {
522 // use staging property, since recording on UI thread
523 mDisplayListData->projectionReceiveIndex = opIndex;
524 }
525 return opIndex;
Romain Guy5ff9df62012-01-23 17:09:05 -0800526}
527
Romain Guy4aa90572010-09-26 18:40:37 -0700528}; // namespace uirenderer
529}; // namespace android