blob: b18eb8fdfd888fc423ac05a02e203a8ef0223c34 [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"
John Reck12f5e342014-11-07 07:53:43 -080026#include "DeferredLayerUpdater.h"
Chet Haase9c1e23b2011-03-24 10:51:31 -070027#include "DisplayListLogBuffer.h"
Chris Craik2af46352012-11-26 18:30:17 -080028#include "DisplayListOp.h"
Romain Guy4aa90572010-09-26 18:40:37 -070029#include "DisplayListRenderer.h"
John Reck113e0822014-03-18 09:22:59 -070030#include "RenderNode.h"
Romain Guy13631f32012-01-30 17:41:55 -080031
Romain Guy4aa90572010-09-26 18:40:37 -070032namespace android {
33namespace uirenderer {
34
Chris Craikcce47eb2014-07-16 15:12:15 -070035DisplayListRenderer::DisplayListRenderer()
36 : mCaches(Caches::getInstance())
Chris Craik8afd0f22014-08-21 17:41:57 -070037 , mDisplayListData(NULL)
Chris Craikcce47eb2014-07-16 15:12:15 -070038 , mTranslateX(0.0f)
39 , mTranslateY(0.0f)
Chris Craik8afd0f22014-08-21 17:41:57 -070040 , mDeferredBarrierType(kBarrier_None)
Chris Craikcce47eb2014-07-16 15:12:15 -070041 , mHighContrastText(false)
42 , mRestoreSaveCount(-1) {
Romain Guy4aa90572010-09-26 18:40:37 -070043}
44
45DisplayListRenderer::~DisplayListRenderer() {
John Reck44fd8d22014-02-26 11:00:11 -080046 LOG_ALWAYS_FATAL_IF(mDisplayListData,
47 "Destroyed a DisplayListRenderer during a record!");
Romain Guy4aa90572010-09-26 18:40:37 -070048}
49
50///////////////////////////////////////////////////////////////////////////////
51// Operations
52///////////////////////////////////////////////////////////////////////////////
53
John Reck44fd8d22014-02-26 11:00:11 -080054DisplayListData* DisplayListRenderer::finishRecording() {
John Reck44fd8d22014-02-26 11:00:11 -080055 mPaintMap.clear();
56 mRegionMap.clear();
57 mPathMap.clear();
58 DisplayListData* data = mDisplayListData;
59 mDisplayListData = 0;
60 return data;
Chet Haase5977baa2011-01-05 18:01:22 -080061}
62
Romain Guy7c25aab2012-10-18 15:05:02 -070063status_t DisplayListRenderer::prepareDirty(float left, float top,
Romain Guy7d7b5492011-01-24 16:33:45 -080064 float right, float bottom, bool opaque) {
John Reck44fd8d22014-02-26 11:00:11 -080065
66 LOG_ALWAYS_FATAL_IF(mDisplayListData,
67 "prepareDirty called a second time during a recording!");
68 mDisplayListData = new DisplayListData();
69
Chris Craik69e5adf2014-08-14 13:34:01 -070070 initializeSaveStack(0, 0, getWidth(), getHeight(), Vector3());
Romain Guy45e4c3d2012-09-11 17:17:07 -070071
Chris Craik8afd0f22014-08-21 17:41:57 -070072 mDeferredBarrierType = kBarrier_InOrder;
Romain Guy45e4c3d2012-09-11 17:17:07 -070073 mDirtyClip = opaque;
Romain Guy27454a42011-01-23 12:01:41 -080074 mRestoreSaveCount = -1;
Romain Guy45e4c3d2012-09-11 17:17:07 -070075
Chet Haase44b2fe32012-06-06 19:03:58 -070076 return DrawGlInfo::kStatusDone; // No invalidate needed at record-time
Romain Guy27454a42011-01-23 12:01:41 -080077}
78
79void DisplayListRenderer::finish() {
Chris Craik8afd0f22014-08-21 17:41:57 -070080 flushRestoreToCount();
81 flushTranslate();
Romain Guyb051e892010-09-28 19:09:36 -070082}
83
Chet Haasedaf98e92011-01-10 14:10:36 -080084void DisplayListRenderer::interrupt() {
Chet Haasedaf98e92011-01-10 14:10:36 -080085}
Romain Guy2b1847e2011-01-26 13:43:01 -080086
Chet Haasedaf98e92011-01-10 14:10:36 -080087void DisplayListRenderer::resume() {
Romain Guy4aa90572010-09-26 18:40:37 -070088}
89
Romain Guy65549432012-03-26 16:45:05 -070090status_t DisplayListRenderer::callDrawGLFunction(Functor *functor, Rect& dirty) {
Romain Guycabfcc12011-03-07 18:06:46 -080091 // Ignore dirty during recording, it matters only when we replay
Chris Craik2af46352012-11-26 18:30:17 -080092 addDrawOp(new (alloc()) DrawFunctorOp(functor));
John Reck09d5cdd2014-07-24 10:36:08 -070093 mDisplayListData->functors.add(functor);
Romain Guy65549432012-03-26 16:45:05 -070094 return DrawGlInfo::kStatusDone; // No invalidate needed at record-time
Chet Haasedaf98e92011-01-10 14:10:36 -080095}
96
Romain Guy4aa90572010-09-26 18:40:37 -070097int DisplayListRenderer::save(int flags) {
Chris Craik2af46352012-11-26 18:30:17 -080098 addStateOp(new (alloc()) SaveOp(flags));
Chris Craik14e51302013-12-30 15:32:54 -080099 return StatefulBaseRenderer::save(flags);
Romain Guy4aa90572010-09-26 18:40:37 -0700100}
101
102void DisplayListRenderer::restore() {
Romain Guy04c9d8c2011-08-25 14:01:48 -0700103 if (mRestoreSaveCount < 0) {
Romain Guy33f6beb2012-02-16 19:24:51 -0800104 restoreToCount(getSaveCount() - 1);
105 return;
Romain Guy04c9d8c2011-08-25 14:01:48 -0700106 }
Romain Guy33f6beb2012-02-16 19:24:51 -0800107
108 mRestoreSaveCount--;
Chris Craik8afd0f22014-08-21 17:41:57 -0700109 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800110 StatefulBaseRenderer::restore();
Romain Guy4aa90572010-09-26 18:40:37 -0700111}
112
113void DisplayListRenderer::restoreToCount(int saveCount) {
Romain Guy27454a42011-01-23 12:01:41 -0800114 mRestoreSaveCount = saveCount;
Chris Craik8afd0f22014-08-21 17:41:57 -0700115 flushTranslate();
Chris Craik14e51302013-12-30 15:32:54 -0800116 StatefulBaseRenderer::restoreToCount(saveCount);
Romain Guy4aa90572010-09-26 18:40:37 -0700117}
118
119int DisplayListRenderer::saveLayer(float left, float top, float right, float bottom,
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500120 const SkPaint* paint, int flags) {
Chris Craik4ace7302014-09-14 15:49:54 -0700121 // force matrix/clip isolation for layer
122 flags |= SkCanvas::kClip_SaveFlag | SkCanvas::kMatrix_SaveFlag;
123
Derek Sollenbergerd44fbe52014-02-05 16:47:00 -0500124 paint = refPaint(paint);
125 addStateOp(new (alloc()) SaveLayerOp(left, top, right, bottom, paint, flags));
Chris Craik14e51302013-12-30 15:32:54 -0800126 return StatefulBaseRenderer::save(flags);
Romain Guy5b3b3522010-10-27 18:57:51 -0700127}
128
Chris Craikb4589422013-12-26 15:13:13 -0800129void DisplayListRenderer::translate(float dx, float dy, float dz) {
130 // ignore dz, not used at defer time
Chris Craik8afd0f22014-08-21 17:41:57 -0700131 mHasDeferredTranslate = true;
Romain Guy33f6beb2012-02-16 19:24:51 -0800132 mTranslateX += dx;
133 mTranslateY += dy;
Chris Craik8afd0f22014-08-21 17:41:57 -0700134 flushRestoreToCount();
Chris Craik14e51302013-12-30 15:32:54 -0800135 StatefulBaseRenderer::translate(dx, dy, dz);
Romain Guy4aa90572010-09-26 18:40:37 -0700136}
137
138void DisplayListRenderer::rotate(float degrees) {
Chris Craik2af46352012-11-26 18:30:17 -0800139 addStateOp(new (alloc()) RotateOp(degrees));
Chris Craik14e51302013-12-30 15:32:54 -0800140 StatefulBaseRenderer::rotate(degrees);
Romain Guy4aa90572010-09-26 18:40:37 -0700141}
142
143void DisplayListRenderer::scale(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800144 addStateOp(new (alloc()) ScaleOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800145 StatefulBaseRenderer::scale(sx, sy);
Romain Guy4aa90572010-09-26 18:40:37 -0700146}
147
Romain Guy807daf72011-01-18 11:19:19 -0800148void DisplayListRenderer::skew(float sx, float sy) {
Chris Craik2af46352012-11-26 18:30:17 -0800149 addStateOp(new (alloc()) SkewOp(sx, sy));
Chris Craik14e51302013-12-30 15:32:54 -0800150 StatefulBaseRenderer::skew(sx, sy);
Romain Guy807daf72011-01-18 11:19:19 -0800151}
152
Derek Sollenberger13908822013-12-10 12:28:58 -0500153void DisplayListRenderer::setMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800154 addStateOp(new (alloc()) SetMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800155 StatefulBaseRenderer::setMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700156}
157
Derek Sollenberger13908822013-12-10 12:28:58 -0500158void DisplayListRenderer::concatMatrix(const SkMatrix& matrix) {
Chris Craik2af46352012-11-26 18:30:17 -0800159 addStateOp(new (alloc()) ConcatMatrixOp(matrix));
Chris Craik14e51302013-12-30 15:32:54 -0800160 StatefulBaseRenderer::concatMatrix(matrix);
Romain Guy4aa90572010-09-26 18:40:37 -0700161}
162
163bool DisplayListRenderer::clipRect(float left, float top, float right, float bottom,
164 SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800165 addStateOp(new (alloc()) ClipRectOp(left, top, right, bottom, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800166 return StatefulBaseRenderer::clipRect(left, top, right, bottom, op);
Romain Guy4aa90572010-09-26 18:40:37 -0700167}
168
Chris Craikd218a922014-01-02 17:13:34 -0800169bool DisplayListRenderer::clipPath(const SkPath* path, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800170 path = refPath(path);
171 addStateOp(new (alloc()) ClipPathOp(path, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800172 return StatefulBaseRenderer::clipPath(path, op);
Romain Guy735738c2012-12-03 12:34:51 -0800173}
174
Chris Craikd218a922014-01-02 17:13:34 -0800175bool DisplayListRenderer::clipRegion(const SkRegion* region, SkRegion::Op op) {
Chris Craik2af46352012-11-26 18:30:17 -0800176 region = refRegion(region);
177 addStateOp(new (alloc()) ClipRegionOp(region, op));
Chris Craikd6b65f62014-01-01 14:45:21 -0800178 return StatefulBaseRenderer::clipRegion(region, op);
Romain Guy735738c2012-12-03 12:34:51 -0800179}
180
Chris Craika7090e02014-06-20 16:01:00 -0700181status_t DisplayListRenderer::drawRenderNode(RenderNode* renderNode, Rect& dirty, int32_t flags) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700182 LOG_ALWAYS_FATAL_IF(!renderNode, "missing rendernode");
183
Romain Guycabfcc12011-03-07 18:06:46 -0800184 // dirty is an out parameter and should not be recorded,
185 // it matters only when replaying the display list
Chris Craikb3cca872014-08-08 18:42:51 -0700186 DrawRenderNodeOp* op = new (alloc()) DrawRenderNodeOp(renderNode, flags, *currentTransform());
Chris Craik8afd0f22014-08-21 17:41:57 -0700187 addRenderNodeOp(op);
Chris Craik2af46352012-11-26 18:30:17 -0800188
Romain Guy65549432012-03-26 16:45:05 -0700189 return DrawGlInfo::kStatusDone;
Romain Guy0fe478e2010-11-08 12:08:41 -0800190}
191
John Reck12f5e342014-11-07 07:53:43 -0800192status_t DisplayListRenderer::drawLayer(DeferredLayerUpdater* layerHandle, float x, float y) {
193 // We ref the DeferredLayerUpdater due to its thread-safe ref-counting
194 // semantics.
195 mDisplayListData->ref(layerHandle);
196 addDrawOp(new (alloc()) DrawLayerOp(layerHandle->backingLayer(), x, y));
Chet Haase48659092012-05-31 15:21:51 -0700197 return DrawGlInfo::kStatusDone;
Romain Guy6c319ca2011-01-11 14:29:25 -0800198}
199
Chris Craik79647502014-08-06 13:42:24 -0700200status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800201 bitmap = refBitmap(bitmap);
202 paint = refPaint(paint);
203
Chris Craik79647502014-08-06 13:42:24 -0700204 addDrawOp(new (alloc()) DrawBitmapOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700205 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700206}
207
Chris Craikd218a922014-01-02 17:13:34 -0800208status_t DisplayListRenderer::drawBitmap(const SkBitmap* bitmap, float srcLeft, float srcTop,
Romain Guy4aa90572010-09-26 18:40:37 -0700209 float srcRight, float srcBottom, float dstLeft, float dstTop,
Chris Craikd218a922014-01-02 17:13:34 -0800210 float dstRight, float dstBottom, const SkPaint* paint) {
Chris Craik79647502014-08-06 13:42:24 -0700211 if (srcLeft == 0 && srcTop == 0
212 && srcRight == bitmap->width() && srcBottom == bitmap->height()
213 && (srcBottom - srcTop == dstBottom - dstTop)
214 && (srcRight - srcLeft == dstRight - dstLeft)) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800215 // transform simple rect to rect drawing case into position bitmap ops, since they merge
Chris Craik79647502014-08-06 13:42:24 -0700216 save(SkCanvas::kMatrix_SaveFlag);
217 translate(dstLeft, dstTop);
218 drawBitmap(bitmap, paint);
219 restore();
220 } else {
221 bitmap = refBitmap(bitmap);
222 paint = refPaint(paint);
Chris Craik527a3aa2013-03-04 10:19:31 -0800223
Chris Craik79647502014-08-06 13:42:24 -0700224 addDrawOp(new (alloc()) DrawBitmapRectOp(bitmap,
225 srcLeft, srcTop, srcRight, srcBottom,
226 dstLeft, dstTop, dstRight, dstBottom, paint));
227 }
Chet Haase48659092012-05-31 15:21:51 -0700228 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700229}
230
Chris Craik79647502014-08-06 13:42:24 -0700231status_t DisplayListRenderer::drawBitmapData(const SkBitmap* bitmap, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800232 bitmap = refBitmapData(bitmap);
233 paint = refPaint(paint);
234
Chris Craik79647502014-08-06 13:42:24 -0700235 addDrawOp(new (alloc()) DrawBitmapDataOp(bitmap, paint));
Chet Haase48659092012-05-31 15:21:51 -0700236 return DrawGlInfo::kStatusDone;
Romain Guye651cc62012-05-14 19:44:40 -0700237}
238
Chris Craikd218a922014-01-02 17:13:34 -0800239status_t DisplayListRenderer::drawBitmapMesh(const SkBitmap* bitmap, int meshWidth, int meshHeight,
240 const float* vertices, const int* colors, const SkPaint* paint) {
Chris Craik0664fef2014-04-11 13:40:05 -0700241 int vertexCount = (meshWidth + 1) * (meshHeight + 1);
Chris Craik2af46352012-11-26 18:30:17 -0800242 bitmap = refBitmap(bitmap);
Chris Craik0664fef2014-04-11 13:40:05 -0700243 vertices = refBuffer<float>(vertices, vertexCount * 2); // 2 floats per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800244 paint = refPaint(paint);
Chris Craik0664fef2014-04-11 13:40:05 -0700245 colors = refBuffer<int>(colors, vertexCount); // 1 color per vertex
Chris Craik2af46352012-11-26 18:30:17 -0800246
247 addDrawOp(new (alloc()) DrawBitmapMeshOp(bitmap, meshWidth, meshHeight,
248 vertices, colors, paint));
Chet Haase48659092012-05-31 15:21:51 -0700249 return DrawGlInfo::kStatusDone;
Romain Guy5a7b4662011-01-20 19:09:30 -0800250}
251
Chris Craikd218a922014-01-02 17:13:34 -0800252status_t DisplayListRenderer::drawPatch(const SkBitmap* bitmap, const Res_png_9patch* patch,
253 float left, float top, float right, float bottom, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800254 bitmap = refBitmap(bitmap);
Romain Guye3b0a012013-06-26 15:45:41 -0700255 patch = refPatch(patch);
Romain Guy16ea8d32013-06-21 11:35:52 -0700256 paint = refPaint(paint);
Chris Craik2af46352012-11-26 18:30:17 -0800257
Romain Guy03c00b52013-06-20 18:30:28 -0700258 addDrawOp(new (alloc()) DrawPatchOp(bitmap, patch, left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700259 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700260}
261
Chet Haase48659092012-05-31 15:21:51 -0700262status_t DisplayListRenderer::drawColor(int color, SkXfermode::Mode mode) {
Chris Craik2af46352012-11-26 18:30:17 -0800263 addDrawOp(new (alloc()) DrawColorOp(color, mode));
Chet Haase48659092012-05-31 15:21:51 -0700264 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700265}
266
Chet Haase48659092012-05-31 15:21:51 -0700267status_t DisplayListRenderer::drawRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800268 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800269 paint = refPaint(paint);
270 addDrawOp(new (alloc()) DrawRectOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700271 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700272}
273
Chet Haase48659092012-05-31 15:21:51 -0700274status_t DisplayListRenderer::drawRoundRect(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800275 float rx, float ry, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800276 paint = refPaint(paint);
277 addDrawOp(new (alloc()) DrawRoundRectOp(left, top, right, bottom, rx, ry, paint));
Chet Haase48659092012-05-31 15:21:51 -0700278 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800279}
280
Jorim Jaggi072707d2014-09-15 17:20:08 +0200281status_t DisplayListRenderer::drawRoundRect(
282 CanvasPropertyPrimitive* left, CanvasPropertyPrimitive* top,
283 CanvasPropertyPrimitive* right, CanvasPropertyPrimitive* bottom,
284 CanvasPropertyPrimitive* rx, CanvasPropertyPrimitive* ry,
285 CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700286 mDisplayListData->ref(left);
287 mDisplayListData->ref(top);
288 mDisplayListData->ref(right);
289 mDisplayListData->ref(bottom);
290 mDisplayListData->ref(rx);
291 mDisplayListData->ref(ry);
292 mDisplayListData->ref(paint);
Jorim Jaggi072707d2014-09-15 17:20:08 +0200293 addDrawOp(new (alloc()) DrawRoundRectPropsOp(&left->value, &top->value,
294 &right->value, &bottom->value, &rx->value, &ry->value, &paint->value));
295 return DrawGlInfo::kStatusDone;
296}
297
Chris Craikd218a922014-01-02 17:13:34 -0800298status_t DisplayListRenderer::drawCircle(float x, float y, float radius, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800299 paint = refPaint(paint);
300 addDrawOp(new (alloc()) DrawCircleOp(x, y, radius, paint));
Chet Haase48659092012-05-31 15:21:51 -0700301 return DrawGlInfo::kStatusDone;
Romain Guy01d58e42011-01-19 21:54:02 -0800302}
303
John Reck52244ff2014-05-01 21:27:37 -0700304status_t DisplayListRenderer::drawCircle(CanvasPropertyPrimitive* x, CanvasPropertyPrimitive* y,
305 CanvasPropertyPrimitive* radius, CanvasPropertyPaint* paint) {
John Reck0e89e2b2014-10-31 14:49:06 -0700306 mDisplayListData->ref(x);
307 mDisplayListData->ref(y);
308 mDisplayListData->ref(radius);
309 mDisplayListData->ref(paint);
John Reck52244ff2014-05-01 21:27:37 -0700310 addDrawOp(new (alloc()) DrawCirclePropsOp(&x->value, &y->value,
311 &radius->value, &paint->value));
312 return DrawGlInfo::kStatusDone;
313}
314
Chet Haase48659092012-05-31 15:21:51 -0700315status_t DisplayListRenderer::drawOval(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800316 const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800317 paint = refPaint(paint);
318 addDrawOp(new (alloc()) DrawOvalOp(left, top, right, bottom, paint));
Chet Haase48659092012-05-31 15:21:51 -0700319 return DrawGlInfo::kStatusDone;
Romain Guyc1cd9ba32011-01-23 14:18:41 -0800320}
321
Chet Haase48659092012-05-31 15:21:51 -0700322status_t DisplayListRenderer::drawArc(float left, float top, float right, float bottom,
Chris Craikd218a922014-01-02 17:13:34 -0800323 float startAngle, float sweepAngle, bool useCenter, const SkPaint* paint) {
Chris Craik544e5242014-07-11 10:58:10 -0700324 if (fabs(sweepAngle) >= 360.0f) {
Chris Craik6ac174b2014-06-17 13:47:05 -0700325 return drawOval(left, top, right, bottom, paint);
326 }
327
Chris Craik2af46352012-11-26 18:30:17 -0800328 paint = refPaint(paint);
329 addDrawOp(new (alloc()) DrawArcOp(left, top, right, bottom,
330 startAngle, sweepAngle, useCenter, paint));
Chet Haase48659092012-05-31 15:21:51 -0700331 return DrawGlInfo::kStatusDone;
Romain Guy8b2f5262011-01-23 16:15:02 -0800332}
333
Chris Craikd218a922014-01-02 17:13:34 -0800334status_t DisplayListRenderer::drawPath(const SkPath* path, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800335 path = refPath(path);
336 paint = refPaint(paint);
Romain Guy33f6beb2012-02-16 19:24:51 -0800337
Chris Craik2af46352012-11-26 18:30:17 -0800338 addDrawOp(new (alloc()) DrawPathOp(path, paint));
Chet Haase48659092012-05-31 15:21:51 -0700339 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700340}
341
Chris Craikd218a922014-01-02 17:13:34 -0800342status_t DisplayListRenderer::drawLines(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800343 points = refBuffer<float>(points, count);
344 paint = refPaint(paint);
345
346 addDrawOp(new (alloc()) DrawLinesOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700347 return DrawGlInfo::kStatusDone;
Romain Guy4aa90572010-09-26 18:40:37 -0700348}
349
Chris Craikd218a922014-01-02 17:13:34 -0800350status_t DisplayListRenderer::drawPoints(const float* points, int count, const SkPaint* paint) {
Chris Craik2af46352012-11-26 18:30:17 -0800351 points = refBuffer<float>(points, count);
352 paint = refPaint(paint);
353
354 addDrawOp(new (alloc()) DrawPointsOp(points, count, paint));
Chet Haase48659092012-05-31 15:21:51 -0700355 return DrawGlInfo::kStatusDone;
Romain Guyed6fcb02011-03-21 13:11:28 -0700356}
357
Chet Haase48659092012-05-31 15:21:51 -0700358status_t DisplayListRenderer::drawTextOnPath(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800359 const SkPath* path, float hOffset, float vOffset, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700360 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800361
Chris Craik2af46352012-11-26 18:30:17 -0800362 text = refText(text, bytesCount);
363 path = refPath(path);
364 paint = refPaint(paint);
365
366 DrawOp* op = new (alloc()) DrawTextOnPathOp(text, bytesCount, count, path,
367 hOffset, vOffset, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800368 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700369 return DrawGlInfo::kStatusDone;
Romain Guy325740f2012-02-24 16:48:34 -0800370}
371
Chet Haase48659092012-05-31 15:21:51 -0700372status_t DisplayListRenderer::drawPosText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800373 const float* positions, const SkPaint* paint) {
Chet Haase48659092012-05-31 15:21:51 -0700374 if (!text || count <= 0) return DrawGlInfo::kStatusDone;
Chris Craik2af46352012-11-26 18:30:17 -0800375
Chris Craik2af46352012-11-26 18:30:17 -0800376 text = refText(text, bytesCount);
377 positions = refBuffer<float>(positions, count * 2);
378 paint = refPaint(paint);
379
380 DrawOp* op = new (alloc()) DrawPosTextOp(text, bytesCount, count, positions, paint);
Romain Guy0f667532013-03-01 14:31:04 -0800381 addDrawOp(op);
Chet Haase48659092012-05-31 15:21:51 -0700382 return DrawGlInfo::kStatusDone;
Romain Guyeb9a5362012-01-17 17:39:26 -0800383}
384
Chris Craikcce47eb2014-07-16 15:12:15 -0700385static void simplifyPaint(int color, SkPaint* paint) {
386 paint->setColor(color);
387 paint->setShader(NULL);
388 paint->setColorFilter(NULL);
389 paint->setLooper(NULL);
390 paint->setStrokeWidth(4 + 0.04 * paint->getTextSize());
391 paint->setStrokeJoin(SkPaint::kRound_Join);
392 paint->setLooper(NULL);
393}
394
Romain Guyc2525952012-07-27 16:41:22 -0700395status_t DisplayListRenderer::drawText(const char* text, int bytesCount, int count,
Chris Craikd218a922014-01-02 17:13:34 -0800396 float x, float y, const float* positions, const SkPaint* paint,
Chris Craik41541822013-05-03 16:35:54 -0700397 float totalAdvance, const Rect& bounds, DrawOpMode drawOpMode) {
Chris Craik527a3aa2013-03-04 10:19:31 -0800398
Chris Craik947eabf2014-08-19 10:21:12 -0700399 if (!text || count <= 0 || paintWillNotDrawText(*paint)) return DrawGlInfo::kStatusDone;
Raph Levien996e57c2012-07-23 15:22:52 -0700400
Chris Craik2af46352012-11-26 18:30:17 -0800401 text = refText(text, bytesCount);
402 positions = refBuffer<float>(positions, count * 2);
Raph Levien996e57c2012-07-23 15:22:52 -0700403
Chris Craikcce47eb2014-07-16 15:12:15 -0700404 if (CC_UNLIKELY(mHighContrastText)) {
405 // high contrast draw path
406 int color = paint->getColor();
407 int channelSum = SkColorGetR(color) + SkColorGetG(color) + SkColorGetB(color);
408 bool darken = channelSum < (128 * 3);
409
410 // outline
411 SkPaint* outlinePaint = copyPaint(paint);
412 simplifyPaint(darken ? SK_ColorWHITE : SK_ColorBLACK, outlinePaint);
413 outlinePaint->setStyle(SkPaint::kStrokeAndFill_Style);
414 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
415 x, y, positions, outlinePaint, totalAdvance, bounds)); // bounds?
416
417 // inner
418 SkPaint* innerPaint = copyPaint(paint);
419 simplifyPaint(darken ? SK_ColorBLACK : SK_ColorWHITE, innerPaint);
420 innerPaint->setStyle(SkPaint::kFill_Style);
421 addDrawOp(new (alloc()) DrawTextOp(text, bytesCount, count,
422 x, y, positions, innerPaint, totalAdvance, bounds));
423 } else {
424 // standard draw path
425 paint = refPaint(paint);
426
427 DrawOp* op = new (alloc()) DrawTextOp(text, bytesCount, count,
428 x, y, positions, paint, totalAdvance, bounds);
429 addDrawOp(op);
430 }
Raph Levien996e57c2012-07-23 15:22:52 -0700431 return DrawGlInfo::kStatusDone;
432}
433
Chris Craikd218a922014-01-02 17:13:34 -0800434status_t DisplayListRenderer::drawRects(const float* rects, int count, const SkPaint* paint) {
Romain Guy672433d2013-01-04 19:05:13 -0800435 if (count <= 0) return DrawGlInfo::kStatusDone;
436
Chris Craik2af46352012-11-26 18:30:17 -0800437 rects = refBuffer<float>(rects, count);
438 paint = refPaint(paint);
439 addDrawOp(new (alloc()) DrawRectsOp(rects, count, paint));
Romain Guy672433d2013-01-04 19:05:13 -0800440 return DrawGlInfo::kStatusDone;
441}
442
Romain Guy5ff9df62012-01-23 17:09:05 -0800443void DisplayListRenderer::resetPaintFilter() {
Chris Craik2af46352012-11-26 18:30:17 -0800444 addStateOp(new (alloc()) ResetPaintFilterOp());
Romain Guy5ff9df62012-01-23 17:09:05 -0800445}
446
447void DisplayListRenderer::setupPaintFilter(int clearBits, int setBits) {
Chris Craik2af46352012-11-26 18:30:17 -0800448 addStateOp(new (alloc()) SetupPaintFilterOp(clearBits, setBits));
449}
450
Chris Craik8afd0f22014-08-21 17:41:57 -0700451void DisplayListRenderer::insertReorderBarrier(bool enableReorder) {
452 flushRestoreToCount();
453 flushTranslate();
454 mDeferredBarrierType = enableReorder ? kBarrier_OutOfOrder : kBarrier_InOrder;
455}
456
457void DisplayListRenderer::flushRestoreToCount() {
Chris Craik2af46352012-11-26 18:30:17 -0800458 if (mRestoreSaveCount >= 0) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700459 addOpAndUpdateChunk(new (alloc()) RestoreToCountOp(mRestoreSaveCount));
Chris Craik2af46352012-11-26 18:30:17 -0800460 mRestoreSaveCount = -1;
461 }
462}
463
Chris Craik8afd0f22014-08-21 17:41:57 -0700464void DisplayListRenderer::flushTranslate() {
465 if (mHasDeferredTranslate) {
Chris Craik2af46352012-11-26 18:30:17 -0800466 if (mTranslateX != 0.0f || mTranslateY != 0.0f) {
Chris Craik8afd0f22014-08-21 17:41:57 -0700467 addOpAndUpdateChunk(new (alloc()) TranslateOp(mTranslateX, mTranslateY));
Chris Craik2af46352012-11-26 18:30:17 -0800468 mTranslateX = mTranslateY = 0.0f;
469 }
Chris Craik8afd0f22014-08-21 17:41:57 -0700470 mHasDeferredTranslate = false;
Chris Craik2af46352012-11-26 18:30:17 -0800471 }
472}
473
Chris Craik8afd0f22014-08-21 17:41:57 -0700474size_t DisplayListRenderer::addOpAndUpdateChunk(DisplayListOp* op) {
475 int insertIndex = mDisplayListData->displayListOps.add(op);
476 if (mDeferredBarrierType != kBarrier_None) {
477 // op is first in new chunk
478 mDisplayListData->chunks.push();
479 DisplayListData::Chunk& newChunk = mDisplayListData->chunks.editTop();
480 newChunk.beginOpIndex = insertIndex;
481 newChunk.endOpIndex = insertIndex + 1;
482 newChunk.reorderChildren = (mDeferredBarrierType == kBarrier_OutOfOrder);
483
484 int nextChildIndex = mDisplayListData->children().size();
485 newChunk.beginChildIndex = newChunk.endChildIndex = nextChildIndex;
486 mDeferredBarrierType = kBarrier_None;
487 } else {
488 // standard case - append to existing chunk
489 mDisplayListData->chunks.editTop().endOpIndex = insertIndex + 1;
490 }
491 return insertIndex;
Chris Craik2af46352012-11-26 18:30:17 -0800492}
493
Chris Craik8afd0f22014-08-21 17:41:57 -0700494size_t DisplayListRenderer::flushAndAddOp(DisplayListOp* op) {
495 flushRestoreToCount();
496 flushTranslate();
497 return addOpAndUpdateChunk(op);
498}
499
500size_t DisplayListRenderer::addStateOp(StateOp* op) {
501 return flushAndAddOp(op);
502}
503
504size_t DisplayListRenderer::addDrawOp(DrawOp* op) {
Chris Craik2af46352012-11-26 18:30:17 -0800505 Rect localBounds;
John Reck3b202512014-06-23 13:13:08 -0700506 if (op->getLocalBounds(localBounds)) {
Chris Craikf0a59072013-11-19 18:00:46 -0800507 bool rejected = quickRejectConservative(localBounds.left, localBounds.top,
Chris Craik2af46352012-11-26 18:30:17 -0800508 localBounds.right, localBounds.bottom);
509 op->setQuickRejected(rejected);
510 }
Chris Craikc1c5f082013-09-11 16:23:37 -0700511
John Reck44fd8d22014-02-26 11:00:11 -0800512 mDisplayListData->hasDrawOps = true;
Chris Craik8afd0f22014-08-21 17:41:57 -0700513 return flushAndAddOp(op);
514}
515
516size_t DisplayListRenderer::addRenderNodeOp(DrawRenderNodeOp* op) {
517 int opIndex = addDrawOp(op);
518 int childIndex = mDisplayListData->addChild(op);
519
520 // update the chunk's child indices
521 DisplayListData::Chunk& chunk = mDisplayListData->chunks.editTop();
522 chunk.endChildIndex = childIndex + 1;
523
524 if (op->renderNode()->stagingProperties().isProjectionReceiver()) {
525 // use staging property, since recording on UI thread
526 mDisplayListData->projectionReceiveIndex = opIndex;
527 }
528 return opIndex;
Romain Guy5ff9df62012-01-23 17:09:05 -0800529}
530
Romain Guy4aa90572010-09-26 18:40:37 -0700531}; // namespace uirenderer
532}; // namespace android