blob: 5892978bfdba33ba1a796076e433ba831ed0a5c4 [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) {
120 paint = refPaint(paint);
121 addStateOp(new (alloc()) SaveLayerOp(left, top, right, bottom, paint, flags));
Chris Craik14e51302013-12-30 15:32:54 -0800122 return StatefulBaseRenderer::save(flags);
Romain Guy5b3b3522010-10-27 18:57:51 -0700123}
124
Chris Craikb4589422013-12-26 15:13:13 -0800125void DisplayListRenderer::translate(float dx, float dy, float dz) {
126 // ignore dz, not used at defer time
Chris Craik8afd0f22014-08-21 17:41:57 -0700127 mHasDeferredTranslate = true;
Romain Guy33f6beb2012-02-16 19:24:51 -0800128 mTranslateX += dx;
129 mTranslateY += dy;
Chris Craik8afd0f22014-08-21 17:41:57 -0700130 flushRestoreToCount();
Chris Craik14e51302013-12-30 15:32:54 -0800131 StatefulBaseRenderer::translate(dx, dy, dz);
Romain Guy4aa90572010-09-26 18:40:37 -0700132}
133
134void DisplayListRenderer::rotate(float degrees) {
Chris Craik2af46352012-11-26 18:30:17 -0800135 addStateOp(new (alloc()) RotateOp(degrees));
Chris Craik14e51302013-12-30 15:32:54 -0800136 StatefulBaseRenderer::rotate(degrees);
Romain Guy4aa90572010-09-26 18:40:37 -0700137}
138
139void DisplayListRenderer::scale(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800140 addStateOp(new (alloc()) ScaleOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800141 StatefulBaseRenderer::scale(sx, sy);
Romain Guy4aa90572010-09-26 18:40:37 -0700142}
143
Romain Guy807daf72011-01-18 11:19:19 -0800144void DisplayListRenderer::skew(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800145 addStateOp(new (alloc()) SkewOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800146 StatefulBaseRenderer::skew(sx, sy);
Romain Guy807daf72011-01-18 11:19:19 -0800147}
148
Derek Sollenberger13908822013-12-10 12:28:58 -0500149void DisplayListRenderer::setMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800150 addStateOp(new (alloc()) SetMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800151 StatefulBaseRenderer::setMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700152}
153
Derek Sollenberger13908822013-12-10 12:28:58 -0500154void DisplayListRenderer::concatMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800155 addStateOp(new (alloc()) ConcatMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800156 StatefulBaseRenderer::concatMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700157}
158
159bool DisplayListRenderer::clipRect(float left, float top, float right, float bottom,
160 SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800161 addStateOp(new (alloc()) ClipRectOp(left, top, right, bottom, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800162 return StatefulBaseRenderer::clipRect(left, top, right, bottom, op);
Romain Guy4aa90572010-09-26 18:40:37 -0700163}
164
Chris Craikd218a922014-01-02 17:13:34 -0800165bool DisplayListRenderer::clipPath(const SkPath* path, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800166 path = refPath(path);
167 addStateOp(new (alloc()) ClipPathOp(path, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800168 return StatefulBaseRenderer::clipPath(path, op);
Romain Guy735738c2012-12-03 12:34:51 -0800169}
170
Chris Craikd218a922014-01-02 17:13:34 -0800171bool DisplayListRenderer::clipRegion(const SkRegion* region, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800172 region = refRegion(region);
173 addStateOp(new (alloc()) ClipRegionOp(region, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800174 return StatefulBaseRenderer::clipRegion(region, op);
Romain Guy735738c2012-12-03 12:34:51 -0800175}
176
Chris Craika7090e02014-06-20 16:01:00 -0700177status_t DisplayListRenderer::drawRenderNode(RenderNode* renderNode, Rect& dirty, int32_t flags) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700178 LOG_ALWAYS_FATAL_IF(!renderNode, "missing rendernode");
179
Romain Guycabfcc12011-03-07 18:06:46 -0800180 // dirty is an out parameter and should not be recorded,
181 // it matters only when replaying the display list
Chris Craikb3cca872014-08-08 18:42:51 -0700182 DrawRenderNodeOp* op = new (alloc()) DrawRenderNodeOp(renderNode, flags, *currentTransform());
Chris Craik8afd0f22014-08-21 17:41:57 -0700183 addRenderNodeOp(op);
Chris Craik2af46352012-11-26 18:30:17 -0800184
Romain Guy65549432012-03-26 16:45:05 -0700185 return DrawGlInfo::kStatusDone;
Romain Guy0fe478e2010-11-08 12:08:41 -0800186}
187
Chris Craika08f95c2013-03-15 17:24:33 -0700188status_t DisplayListRenderer::drawLayer(Layer* layer, float x, float y) {
Romain Guyce4a7df2013-03-28 11:32:33 -0700189 layer = refLayer(layer);
Chris Craika08f95c2013-03-15 17:24:33 -0700190 addDrawOp(new (alloc()) DrawLayerOp(layer, x, y));
Chet Haase48659092012-05-31 15:21:51 -0700191 return DrawGlInfo::kStatusDone;
Romain Guy6c319ca2011-01-11 14:29:25 -0800192}
193
Chris Craik79647502014-08-06 13:42:24 -0700194status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800195 bitmap = refBitmap(bitmap);
196 paint = refPaint(paint);
197
Chris Craik79647502014-08-06 13:42:24 -0700198 addDrawOp(new (alloc()) DrawBitmapOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700199 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700200}
201
Chris Craikd218a922014-01-02 17:13:34 -0800202status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, float srcLeft, float srcTop,
Romain Guy4aa90572010-09-26 18:40:37 -0700203 float srcRight, float srcBottom, float dstLeft, float dstTop,
Chris Craikd218a922014-01-02 17:13:34 -0800204 float dstRight, float dstBottom, const SkPaint* paint) {
Chris Craik79647502014-08-06 13:42:24 -0700205 if (srcLeft == 0 && srcTop == 0
206 && srcRight == bitmap->width() && srcBottom == bitmap->height()
207 && (srcBottom - srcTop == dstBottom - dstTop)
208 && (srcRight - srcLeft == dstRight - dstLeft)) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800209 // transform simple rect to rect drawing case into position bitmap ops, since they merge
Chris Craik79647502014-08-06 13:42:24 -0700210 save(SkCanvas::kMatrix_SaveFlag);
211 translate(dstLeft, dstTop);
212 drawBitmap(bitmap, paint);
213 restore();
214 } else {
215 bitmap = refBitmap(bitmap);
216 paint = refPaint(paint);
Chris Craik527a3aa2013-03-04 10:19:31 -0800217
Chris Craik79647502014-08-06 13:42:24 -0700218 addDrawOp(new (alloc()) DrawBitmapRectOp(bitmap,
219 srcLeft, srcTop, srcRight, srcBottom,
220 dstLeft, dstTop, dstRight, dstBottom, paint));
221 }
Chet Haase48659092012-05-31 15:21:51 -0700222 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700223}
224
Chris Craik79647502014-08-06 13:42:24 -0700225status_t DisplayListRenderer::drawBitmapData(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800226 bitmap = refBitmapData(bitmap);
227 paint = refPaint(paint);
228
Chris Craik79647502014-08-06 13:42:24 -0700229 addDrawOp(new (alloc()) DrawBitmapDataOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700230 return DrawGlInfo::kStatusDone;
Romain Guye651cc62012-05-14 19:44:40 -0700231}
232
Chris Craikd218a922014-01-02 17:13:34 -0800233status_t DisplayListRenderer::drawBitmapMesh(const SkBitmap* bitmap, int meshWidth, int meshHeight,
234 const float* vertices, const int* colors, const SkPaint* paint) {
Chris Craik0664fef2014-04-11 13:40:05 -0700235 int vertexCount = (meshWidth + 1) * (meshHeight + 1);
Chris Craik2af46352012-11-26 18:30:17 -0800236 bitmap = refBitmap(bitmap);
Chris Craik0664fef2014-04-11 13:40:05 -0700237 vertices = refBuffer<float>(vertices, vertexCount * 2); // 2 floats per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800238 paint = refPaint(paint);
Chris Craik0664fef2014-04-11 13:40:05 -0700239 colors = refBuffer<int>(colors, vertexCount); // 1 color per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800240
241 addDrawOp(new (alloc()) DrawBitmapMeshOp(bitmap, meshWidth, meshHeight,
242 vertices, colors, paint));
Chet Haase48659092012-05-31 15:21:51 -0700243 return DrawGlInfo::kStatusDone;
Romain Guy5a7b4662011-01-20 19:09:30 -0800244}
245
Chris Craikd218a922014-01-02 17:13:34 -0800246status_t DisplayListRenderer::drawPatch(const SkBitmap* bitmap, const Res_png_9patch* patch,
247 float left, float top, float right, float bottom, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800248 bitmap = refBitmap(bitmap);
Romain Guye3b0a012013-06-26 15:45:41 -0700249 patch = refPatch(patch);
Romain Guy16ea8d32013-06-21 11:35:52 -0700250 paint = refPaint(paint);
Chris Craik2af46352012-11-26 18:30:17 -0800251
Romain Guy03c00b52013-06-20 18:30:28 -0700252 addDrawOp(new (alloc()) DrawPatchOp(bitmap, patch, left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700253 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700254}
255
Chet Haase48659092012-05-31 15:21:51 -0700256status_t DisplayListRenderer::drawColor(int color, SkXfermode::Mode mode) {
Chris Craik2af46352012-11-26 18:30:17 -0800257 addDrawOp(new (alloc()) DrawColorOp(color, mode));
Chet Haase48659092012-05-31 15:21:51 -0700258 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700259}
260
Chet Haase48659092012-05-31 15:21:51 -0700261status_t DisplayListRenderer::drawRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800262 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800263 paint = refPaint(paint);
264 addDrawOp(new (alloc()) DrawRectOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700265 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700266}
267
Chet Haase48659092012-05-31 15:21:51 -0700268status_t DisplayListRenderer::drawRoundRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800269 float rx, float ry, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800270 paint = refPaint(paint);
271 addDrawOp(new (alloc()) DrawRoundRectOp(left, top, right, bottom, rx, ry, paint));
Chet Haase48659092012-05-31 15:21:51 -0700272 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800273}
274
Chris Craikd218a922014-01-02 17:13:34 -0800275status_t DisplayListRenderer::drawCircle(float x, float y, float radius, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800276 paint = refPaint(paint);
277 addDrawOp(new (alloc()) DrawCircleOp(x, y, radius, paint));
Chet Haase48659092012-05-31 15:21:51 -0700278 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800279}
280
John Reck52244ff2014-05-01 21:27:37 -0700281status_t DisplayListRenderer::drawCircle(CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y,
282 CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) {
283 mDisplayListData->refProperty(x);
284 mDisplayListData->refProperty(y);
285 mDisplayListData->refProperty(radius);
286 mDisplayListData->refProperty(paint);
287 addDrawOp(new (alloc()) DrawCirclePropsOp(&x->value, &y->value,
288 &radius->value, &paint->value));
289 return DrawGlInfo::kStatusDone;
290}
291
Chet Haase48659092012-05-31 15:21:51 -0700292status_t DisplayListRenderer::drawOval(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800293 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800294 paint = refPaint(paint);
295 addDrawOp(new (alloc()) DrawOvalOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700296 return DrawGlInfo::kStatusDone;
Romain Guyc1cd9ba32011-01-23 14:18:41 -0800297}
298
Chet Haase48659092012-05-31 15:21:51 -0700299status_t DisplayListRenderer::drawArc(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800300 float startAngle, float sweepAngle, bool useCenter, const SkPaint* paint) {
Chris Craik544e5242014-07-11 10:58:10 -0700301 if (fabs(sweepAngle) >= 360.0f) {
Chris Craik6ac174b2014-06-17 13:47:05 -0700302 return drawOval(left, top, right, bottom, paint);
303 }
304
Chris Craik2af46352012-11-26 18:30:17 -0800305 paint = refPaint(paint);
306 addDrawOp(new (alloc()) DrawArcOp(left, top, right, bottom,
307 startAngle, sweepAngle, useCenter, paint));
Chet Haase48659092012-05-31 15:21:51 -0700308 return DrawGlInfo::kStatusDone;
Romain Guy8b2f5262011-01-23 16:15:02 -0800309}
310
Chris Craikd218a922014-01-02 17:13:34 -0800311status_t DisplayListRenderer::drawPath(const SkPath* path, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800312 path = refPath(path);
313 paint = refPaint(paint);
Romain Guy33f6beb2012-02-16 19:24:51 -0800314
Chris Craik2af46352012-11-26 18:30:17 -0800315 addDrawOp(new (alloc()) DrawPathOp(path, paint));
Chet Haase48659092012-05-31 15:21:51 -0700316 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700317}
318
Chris Craikd218a922014-01-02 17:13:34 -0800319status_t DisplayListRenderer::drawLines(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800320 points = refBuffer<float>(points, count);
321 paint = refPaint(paint);
322
323 addDrawOp(new (alloc()) DrawLinesOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700324 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700325}
326
Chris Craikd218a922014-01-02 17:13:34 -0800327status_t DisplayListRenderer::drawPoints(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800328 points = refBuffer<float>(points, count);
329 paint = refPaint(paint);
330
331 addDrawOp(new (alloc()) DrawPointsOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700332 return DrawGlInfo::kStatusDone;
Romain Guyed6fcb02011-03-21 13:11:28 -0700333}
334
Chet Haase48659092012-05-31 15:21:51 -0700335status_t DisplayListRenderer::drawTextOnPath(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800336 const SkPath* path, float hOffset, float vOffset, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700337 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800338
Chris Craik2af46352012-11-26 18:30:17 -0800339 text = refText(text, bytesCount);
340 path = refPath(path);
341 paint = refPaint(paint);
342
343 DrawOp* op = new (alloc()) DrawTextOnPathOp(text, bytesCount, count, path,
344 hOffset, vOffset, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800345 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700346 return DrawGlInfo::kStatusDone;
Romain Guy325740f2012-02-24 16:48:34 -0800347}
348
Chet Haase48659092012-05-31 15:21:51 -0700349status_t DisplayListRenderer::drawPosText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800350 const float* positions, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700351 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800352
Chris Craik2af46352012-11-26 18:30:17 -0800353 text = refText(text, bytesCount);
354 positions = refBuffer<float>(positions, count * 2);
355 paint = refPaint(paint);
356
357 DrawOp* op = new (alloc()) DrawPosTextOp(text, bytesCount, count, positions, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800358 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700359 return DrawGlInfo::kStatusDone;
Romain Guyeb9a5362012-01-17 17:39:26 -0800360}
361
Chris Craikcce47eb2014-07-16 15:12:15 -0700362static void simplifyPaint(int color, SkPaint* paint) {
363 paint->setColor(color);
364 paint->setShader(NULL);
365 paint->setColorFilter(NULL);
366 paint->setLooper(NULL);
367 paint->setStrokeWidth(4 + 0.04 * paint->getTextSize());
368 paint->setStrokeJoin(SkPaint::kRound_Join);
369 paint->setLooper(NULL);
370}
371
Romain Guyc2525952012-07-27 16:41:22 -0700372status_t DisplayListRenderer::drawText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800373 float x, float y, const float* positions, const SkPaint* paint,
Chris Craik41541822013-05-03 16:35:54 -0700374 float totalAdvance, const Rect& bounds, DrawOpMode drawOpMode) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800375
Chris Craik947eabf2014-08-19 10:21:12 -0700376 if (!text || count <= 0 || paintWillNotDrawText(*paint)) return DrawGlInfo::kStatusDone;
Raph Levien996e57c2012-07-23 15:22:52 -0700377
Chris Craik2af46352012-11-26 18:30:17 -0800378 text = refText(text, bytesCount);
379 positions = refBuffer<float>(positions, count * 2);
Raph Levien996e57c2012-07-23 15:22:52 -0700380
Chris Craikcce47eb2014-07-16 15:12:15 -0700381 if (CC_UNLIKELY(mHighContrastText)) {
382 // high contrast draw path
383 int color = paint->getColor();
384 int channelSum = SkColorGetR(color) + SkColorGetG(color) + SkColorGetB(color);
385 bool darken = channelSum < (128 * 3);
386
387 // outline
388 SkPaint* outlinePaint = copyPaint(paint);
389 simplifyPaint(darken ? SK_ColorWHITE : SK_ColorBLACK, outlinePaint);
390 outlinePaint->setStyle(SkPaint::kStrokeAndFill_Style);
391 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
392 x, y, positions, outlinePaint, totalAdvance, bounds)); // bounds?
393
394 // inner
395 SkPaint* innerPaint = copyPaint(paint);
396 simplifyPaint(darken ? SK_ColorBLACK : SK_ColorWHITE, innerPaint);
397 innerPaint->setStyle(SkPaint::kFill_Style);
398 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
399 x, y, positions, innerPaint, totalAdvance, bounds));
400 } else {
401 // standard draw path
402 paint = refPaint(paint);
403
404 DrawOp* op = new (alloc()) DrawTextOp(text, bytesCount, count,
405 x, y, positions, paint, totalAdvance, bounds);
406 addDrawOp(op);
407 }
Raph Levien996e57c2012-07-23 15:22:52 -0700408 return DrawGlInfo::kStatusDone;
409}
410
Chris Craikd218a922014-01-02 17:13:34 -0800411status_t DisplayListRenderer::drawRects(const float* rects, int count, const SkPaint* paint) {
Romain Guy672433d2013-01-04 19:05:13 -0800412 if (count <= 0) return DrawGlInfo::kStatusDone;
413
Chris Craik2af46352012-11-26 18:30:17 -0800414 rects = refBuffer<float>(rects, count);
415 paint = refPaint(paint);
416 addDrawOp(new (alloc()) DrawRectsOp(rects, count, paint));
Romain Guy672433d2013-01-04 19:05:13 -0800417 return DrawGlInfo::kStatusDone;
418}
419
Romain Guy5ff9df62012-01-23 17:09:05 -0800420void DisplayListRenderer::resetPaintFilter() {
Chris Craik2af46352012-11-26 18:30:17 -0800421 addStateOp(new (alloc()) ResetPaintFilterOp());
Romain Guy5ff9df62012-01-23 17:09:05 -0800422}
423
424void DisplayListRenderer::setupPaintFilter(int clearBits, int setBits) {
Chris Craik2af46352012-11-26 18:30:17 -0800425 addStateOp(new (alloc()) SetupPaintFilterOp(clearBits, setBits));
426}
427
Chris Craik8afd0f22014-08-21 17:41:57 -0700428void DisplayListRenderer::insertReorderBarrier(bool enableReorder) {
429 flushRestoreToCount();
430 flushTranslate();
431 mDeferredBarrierType = enableReorder ? kBarrier_OutOfOrder : kBarrier_InOrder;
432}
433
434void DisplayListRenderer::flushRestoreToCount() {
Chris Craik2af46352012-11-26 18:30:17 -0800435 if (mRestoreSaveCount >= 0) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700436 addOpAndUpdateChunk(new (alloc()) RestoreToCountOp(mRestoreSaveCount));
Chris Craik2af46352012-11-26 18:30:17 -0800437 mRestoreSaveCount = -1;
438 }
439}
440
Chris Craik8afd0f22014-08-21 17:41:57 -0700441void DisplayListRenderer::flushTranslate() {
442 if (mHasDeferredTranslate) {
Chris Craik2af46352012-11-26 18:30:17 -0800443 if (mTranslateX != 0.0f || mTranslateY != 0.0f) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700444 addOpAndUpdateChunk(new (alloc()) TranslateOp(mTranslateX, mTranslateY));
Chris Craik2af46352012-11-26 18:30:17 -0800445 mTranslateX = mTranslateY = 0.0f;
446 }
Chris Craik8afd0f22014-08-21 17:41:57 -0700447 mHasDeferredTranslate = false;
Chris Craik2af46352012-11-26 18:30:17 -0800448 }
449}
450
Chris Craik8afd0f22014-08-21 17:41:57 -0700451size_t DisplayListRenderer::addOpAndUpdateChunk(DisplayListOp* op) {
452 int insertIndex = mDisplayListData->displayListOps.add(op);
453 if (mDeferredBarrierType != kBarrier_None) {
454 // op is first in new chunk
455 mDisplayListData->chunks.push();
456 DisplayListData::Chunk& newChunk = mDisplayListData->chunks.editTop();
457 newChunk.beginOpIndex = insertIndex;
458 newChunk.endOpIndex = insertIndex + 1;
459 newChunk.reorderChildren = (mDeferredBarrierType == kBarrier_OutOfOrder);
460
461 int nextChildIndex = mDisplayListData->children().size();
462 newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex;
463 mDeferredBarrierType = kBarrier_None;
464 } else {
465 // standard case - append to existing chunk
466 mDisplayListData->chunks.editTop().endOpIndex = insertIndex + 1;
467 }
468 return insertIndex;
Chris Craik2af46352012-11-26 18:30:17 -0800469}
470
Chris Craik8afd0f22014-08-21 17:41:57 -0700471size_t DisplayListRenderer::flushAndAddOp(DisplayListOp* op) {
472 flushRestoreToCount();
473 flushTranslate();
474 return addOpAndUpdateChunk(op);
475}
476
477size_t DisplayListRenderer::addStateOp(StateOp* op) {
478 return flushAndAddOp(op);
479}
480
481size_t DisplayListRenderer::addDrawOp(DrawOp* op) {
Chris Craik2af46352012-11-26 18:30:17 -0800482 Rect localBounds;
John Reck3b202512014-06-23 13:13:08 -0700483 if (op->getLocalBounds(localBounds)) {
Chris Craikf0a59072013-11-19 18:00:46 -0800484 bool rejected = quickRejectConservative(localBounds.left, localBounds.top,
Chris Craik2af46352012-11-26 18:30:17 -0800485 localBounds.right, localBounds.bottom);
486 op->setQuickRejected(rejected);
487 }
Chris Craikc1c5f082013-09-11 16:23:37 -0700488
John Reck44fd8d22014-02-26 11:00:11 -0800489 mDisplayListData->hasDrawOps = true;
Chris Craik8afd0f22014-08-21 17:41:57 -0700490 return flushAndAddOp(op);
491}
492
493size_t DisplayListRenderer::addRenderNodeOp(DrawRenderNodeOp* op) {
494 int opIndex = addDrawOp(op);
495 int childIndex = mDisplayListData->addChild(op);
496
497 // update the chunk's child indices
498 DisplayListData::Chunk& chunk = mDisplayListData->chunks.editTop();
499 chunk.endChildIndex = childIndex + 1;
500
501 if (op->renderNode()->stagingProperties().isProjectionReceiver()) {
502 // use staging property, since recording on UI thread
503 mDisplayListData->projectionReceiveIndex = opIndex;
504 }
505 return opIndex;
Romain Guy5ff9df62012-01-23 17:09:05 -0800506}
507
Romain Guy4aa90572010-09-26 18:40:37 -0700508}; // namespace uirenderer
509}; // namespace android