1 /* 2 * Copyright (C) 2015 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 #ifndef ANDROID_HWUI_RECORDED_OP_H 18 #define ANDROID_HWUI_RECORDED_OP_H 19 20 #include "RecordedOp.h" 21 #include "font/FontUtil.h" 22 #include "Matrix.h" 23 #include "Rect.h" 24 #include "RenderNode.h" 25 #include "TessellationCache.h" 26 #include "utils/LinearAllocator.h" 27 #include "Vector.h" 28 29 #include <androidfw/ResourceTypes.h> 30 #include <SkXfermode.h> 31 32 class SkBitmap; 33 class SkPaint; 34 35 namespace android { 36 namespace uirenderer { 37 38 struct ClipBase; 39 class OffscreenBuffer; 40 class RenderNode; 41 struct Vertex; 42 43 namespace VectorDrawable { 44 class Tree; 45 } 46 47 /** 48 * Authoritative op list, used for generating the op ID enum, ID based LUTS, and 49 * the functions to which they dispatch. Parameter macros are executed for each op, 50 * in order, based on the op's type. 51 * 52 * There are 4 types of op, which defines dispatch/LUT capability: 53 * 54 * | DisplayList | Render | Merge | 55 * -------------|-------------|-------------|-------------| 56 * PRE RENDER | Yes | | | 57 * RENDER ONLY | | Yes | | 58 * UNMERGEABLE | Yes | Yes | | 59 * MERGEABLE | Yes | Yes | Yes | 60 * 61 * PRE RENDER - These ops are recorded into DisplayLists, but can't be directly rendered. This 62 * may be because they need to be transformed into other op types (e.g. CirclePropsOp), 63 * be traversed to access multiple renderable ops within (e.g. RenderNodeOp), or because they 64 * modify renderbuffer lifecycle, instead of directly rendering content (the various LayerOps). 65 * 66 * RENDER ONLY - These ops cannot be recorded into DisplayLists, and are instead implicitly 67 * constructed from other commands/RenderNode properties. They cannot be merged. 68 * 69 * UNMERGEABLE - These ops can be recorded into DisplayLists and rendered directly, but do not 70 * support merged rendering. 71 * 72 * MERGEABLE - These ops can be recorded into DisplayLists and rendered individually, or merged 73 * under certain circumstances. 74 */ 75 #define MAP_OPS_BASED_ON_TYPE(PRE_RENDER_OP_FN, RENDER_ONLY_OP_FN, UNMERGEABLE_OP_FN, MERGEABLE_OP_FN) \ 76 PRE_RENDER_OP_FN(RenderNodeOp) \ 77 PRE_RENDER_OP_FN(CirclePropsOp) \ 78 PRE_RENDER_OP_FN(RoundRectPropsOp) \ 79 PRE_RENDER_OP_FN(BeginLayerOp) \ 80 PRE_RENDER_OP_FN(EndLayerOp) \ 81 PRE_RENDER_OP_FN(BeginUnclippedLayerOp) \ 82 PRE_RENDER_OP_FN(EndUnclippedLayerOp) \ 83 PRE_RENDER_OP_FN(VectorDrawableOp) \ 84 \ 85 RENDER_ONLY_OP_FN(ShadowOp) \ 86 RENDER_ONLY_OP_FN(LayerOp) \ 87 RENDER_ONLY_OP_FN(CopyToLayerOp) \ 88 RENDER_ONLY_OP_FN(CopyFromLayerOp) \ 89 \ 90 UNMERGEABLE_OP_FN(ArcOp) \ 91 UNMERGEABLE_OP_FN(BitmapMeshOp) \ 92 UNMERGEABLE_OP_FN(BitmapRectOp) \ 93 UNMERGEABLE_OP_FN(ColorOp) \ 94 UNMERGEABLE_OP_FN(FunctorOp) \ 95 UNMERGEABLE_OP_FN(LinesOp) \ 96 UNMERGEABLE_OP_FN(OvalOp) \ 97 UNMERGEABLE_OP_FN(PathOp) \ 98 UNMERGEABLE_OP_FN(PointsOp) \ 99 UNMERGEABLE_OP_FN(RectOp) \ 100 UNMERGEABLE_OP_FN(RoundRectOp) \ 101 UNMERGEABLE_OP_FN(SimpleRectsOp) \ 102 UNMERGEABLE_OP_FN(TextOnPathOp) \ 103 UNMERGEABLE_OP_FN(TextureLayerOp) \ 104 \ 105 MERGEABLE_OP_FN(BitmapOp) \ 106 MERGEABLE_OP_FN(PatchOp) \ 107 MERGEABLE_OP_FN(TextOp) 108 109 /** 110 * LUT generators, which will insert nullptr for unsupported ops 111 */ 112 #define NULLPTR_OP_FN(Type) nullptr, 113 114 #define BUILD_DEFERRABLE_OP_LUT(OP_FN) \ 115 { MAP_OPS_BASED_ON_TYPE(OP_FN, NULLPTR_OP_FN, OP_FN, OP_FN) } 116 117 #define BUILD_MERGEABLE_OP_LUT(OP_FN) \ 118 { MAP_OPS_BASED_ON_TYPE(NULLPTR_OP_FN, NULLPTR_OP_FN, NULLPTR_OP_FN, OP_FN) } 119 120 #define BUILD_RENDERABLE_OP_LUT(OP_FN) \ 121 { MAP_OPS_BASED_ON_TYPE(NULLPTR_OP_FN, OP_FN, OP_FN, OP_FN) } 122 123 #define BUILD_FULL_OP_LUT(OP_FN) \ 124 { MAP_OPS_BASED_ON_TYPE(OP_FN, OP_FN, OP_FN, OP_FN) } 125 126 /** 127 * Op mapping functions, which skip unsupported ops. 128 * 129 * Note: Do not use for LUTS, since these do not preserve ID order. 130 */ 131 #define NULL_OP_FN(Type) 132 133 #define MAP_DEFERRABLE_OPS(OP_FN) \ 134 MAP_OPS_BASED_ON_TYPE(OP_FN, NULL_OP_FN, OP_FN, OP_FN) 135 136 #define MAP_MERGEABLE_OPS(OP_FN) \ 137 MAP_OPS_BASED_ON_TYPE(NULL_OP_FN, NULL_OP_FN, NULL_OP_FN, OP_FN) 138 139 #define MAP_RENDERABLE_OPS(OP_FN) \ 140 MAP_OPS_BASED_ON_TYPE(NULL_OP_FN, OP_FN, OP_FN, OP_FN) 141 142 // Generate OpId enum 143 #define IDENTITY_FN(Type) Type, 144 namespace RecordedOpId { 145 enum { 146 MAP_OPS_BASED_ON_TYPE(IDENTITY_FN, IDENTITY_FN, IDENTITY_FN, IDENTITY_FN) 147 Count, 148 }; 149 } 150 static_assert(RecordedOpId::RenderNodeOp == 0, 151 "First index must be zero for LUTs to work"); 152 153 #define BASE_PARAMS const Rect& unmappedBounds, const Matrix4& localMatrix, const ClipBase* localClip, const SkPaint* paint 154 #define BASE_PARAMS_PAINTLESS const Rect& unmappedBounds, const Matrix4& localMatrix, const ClipBase* localClip 155 #define SUPER(Type) RecordedOp(RecordedOpId::Type, unmappedBounds, localMatrix, localClip, paint) 156 #define SUPER_PAINTLESS(Type) RecordedOp(RecordedOpId::Type, unmappedBounds, localMatrix, localClip, nullptr) 157 158 struct RecordedOp { 159 /* ID from RecordedOpId - generally used for jumping into function tables */ 160 const int opId; 161 162 /* bounds in *local* space, without accounting for DisplayList transformation, or stroke */ 163 const Rect unmappedBounds; 164 165 /* transform in recording space (vs DisplayList origin) */ 166 const Matrix4 localMatrix; 167 168 /* clip in recording space - nullptr if not clipped */ 169 const ClipBase* localClip; 170 171 /* optional paint, stored in base object to simplify merging logic */ 172 const SkPaint* paint; 173 protected: RecordedOpRecordedOp174 RecordedOp(unsigned int opId, BASE_PARAMS) 175 : opId(opId) 176 , unmappedBounds(unmappedBounds) 177 , localMatrix(localMatrix) 178 , localClip(localClip) 179 , paint(paint) {} 180 }; 181 182 struct RenderNodeOp : RecordedOp { RenderNodeOpRenderNodeOp183 RenderNodeOp(BASE_PARAMS_PAINTLESS, RenderNode* renderNode) 184 : SUPER_PAINTLESS(RenderNodeOp) 185 , renderNode(renderNode) {} 186 RenderNode * renderNode; // not const, since drawing modifies it 187 188 /** 189 * Holds the transformation between the projection surface ViewGroup and this RenderNode 190 * drawing instance. Represents any translations / transformations done within the drawing of 191 * the compositing ancestor ViewGroup's draw, before the draw of the View represented by this 192 * DisplayList draw instance. 193 * 194 * Note: doesn't include transformation within the RenderNode, or its properties. 195 */ 196 Matrix4 transformFromCompositingAncestor; 197 bool skipInOrderDraw = false; 198 }; 199 200 //////////////////////////////////////////////////////////////////////////////////////////////////// 201 // Standard Ops 202 //////////////////////////////////////////////////////////////////////////////////////////////////// 203 204 struct ArcOp : RecordedOp { ArcOpArcOp205 ArcOp(BASE_PARAMS, float startAngle, float sweepAngle, bool useCenter) 206 : SUPER(ArcOp) 207 , startAngle(startAngle) 208 , sweepAngle(sweepAngle) 209 , useCenter(useCenter) {} 210 const float startAngle; 211 const float sweepAngle; 212 const bool useCenter; 213 }; 214 215 struct BitmapOp : RecordedOp { BitmapOpBitmapOp216 BitmapOp(BASE_PARAMS, const SkBitmap* bitmap) 217 : SUPER(BitmapOp) 218 , bitmap(bitmap) {} 219 const SkBitmap* bitmap; 220 // TODO: asset atlas/texture id lookup? 221 }; 222 223 struct BitmapMeshOp : RecordedOp { BitmapMeshOpBitmapMeshOp224 BitmapMeshOp(BASE_PARAMS, const SkBitmap* bitmap, int meshWidth, int meshHeight, 225 const float* vertices, const int* colors) 226 : SUPER(BitmapMeshOp) 227 , bitmap(bitmap) 228 , meshWidth(meshWidth) 229 , meshHeight(meshHeight) 230 , vertices(vertices) 231 , colors(colors) {} 232 const SkBitmap* bitmap; 233 const int meshWidth; 234 const int meshHeight; 235 const float* vertices; 236 const int* colors; 237 }; 238 239 struct BitmapRectOp : RecordedOp { BitmapRectOpBitmapRectOp240 BitmapRectOp(BASE_PARAMS, const SkBitmap* bitmap, const Rect& src) 241 : SUPER(BitmapRectOp) 242 , bitmap(bitmap) 243 , src(src) {} 244 const SkBitmap* bitmap; 245 const Rect src; 246 }; 247 248 struct CirclePropsOp : RecordedOp { CirclePropsOpCirclePropsOp249 CirclePropsOp(const Matrix4& localMatrix, const ClipBase* localClip, const SkPaint* paint, 250 float* x, float* y, float* radius) 251 : RecordedOp(RecordedOpId::CirclePropsOp, Rect(), localMatrix, localClip, paint) 252 , x(x) 253 , y(y) 254 , radius(radius) {} 255 const float* x; 256 const float* y; 257 const float* radius; 258 }; 259 260 struct ColorOp : RecordedOp { 261 // Note: unbounded op that will fillclip, so no bounds/matrix needed ColorOpColorOp262 ColorOp(const ClipBase* localClip, int color, SkXfermode::Mode mode) 263 : RecordedOp(RecordedOpId::ColorOp, Rect(), Matrix4::identity(), localClip, nullptr) 264 , color(color) 265 , mode(mode) {} 266 const int color; 267 const SkXfermode::Mode mode; 268 }; 269 270 struct FunctorOp : RecordedOp { 271 // Note: undefined record-time bounds, since this op fills the clip 272 // TODO: explicitly define bounds FunctorOpFunctorOp273 FunctorOp(const Matrix4& localMatrix, const ClipBase* localClip, Functor* functor) 274 : RecordedOp(RecordedOpId::FunctorOp, Rect(), localMatrix, localClip, nullptr) 275 , functor(functor) {} 276 Functor* functor; 277 }; 278 279 struct LinesOp : RecordedOp { LinesOpLinesOp280 LinesOp(BASE_PARAMS, const float* points, const int floatCount) 281 : SUPER(LinesOp) 282 , points(points) 283 , floatCount(floatCount) {} 284 const float* points; 285 const int floatCount; 286 }; 287 288 struct OvalOp : RecordedOp { OvalOpOvalOp289 OvalOp(BASE_PARAMS) 290 : SUPER(OvalOp) {} 291 }; 292 293 struct PatchOp : RecordedOp { PatchOpPatchOp294 PatchOp(BASE_PARAMS, const SkBitmap* bitmap, const Res_png_9patch* patch) 295 : SUPER(PatchOp) 296 , bitmap(bitmap) 297 , patch(patch) {} 298 const SkBitmap* bitmap; 299 const Res_png_9patch* patch; 300 }; 301 302 struct PathOp : RecordedOp { PathOpPathOp303 PathOp(BASE_PARAMS, const SkPath* path) 304 : SUPER(PathOp) 305 , path(path) {} 306 const SkPath* path; 307 }; 308 309 struct PointsOp : RecordedOp { PointsOpPointsOp310 PointsOp(BASE_PARAMS, const float* points, const int floatCount) 311 : SUPER(PointsOp) 312 , points(points) 313 , floatCount(floatCount) {} 314 const float* points; 315 const int floatCount; 316 }; 317 318 struct RectOp : RecordedOp { RectOpRectOp319 RectOp(BASE_PARAMS) 320 : SUPER(RectOp) {} 321 }; 322 323 struct RoundRectOp : RecordedOp { RoundRectOpRoundRectOp324 RoundRectOp(BASE_PARAMS, float rx, float ry) 325 : SUPER(RoundRectOp) 326 , rx(rx) 327 , ry(ry) {} 328 const float rx; 329 const float ry; 330 }; 331 332 struct RoundRectPropsOp : RecordedOp { RoundRectPropsOpRoundRectPropsOp333 RoundRectPropsOp(const Matrix4& localMatrix, const ClipBase* localClip, const SkPaint* paint, 334 float* left, float* top, float* right, float* bottom, float *rx, float *ry) 335 : RecordedOp(RecordedOpId::RoundRectPropsOp, Rect(), localMatrix, localClip, paint) 336 , left(left) 337 , top(top) 338 , right(right) 339 , bottom(bottom) 340 , rx(rx) 341 , ry(ry) {} 342 const float* left; 343 const float* top; 344 const float* right; 345 const float* bottom; 346 const float* rx; 347 const float* ry; 348 }; 349 350 struct VectorDrawableOp : RecordedOp { VectorDrawableOpVectorDrawableOp351 VectorDrawableOp(VectorDrawable::Tree* tree, BASE_PARAMS_PAINTLESS) 352 : SUPER_PAINTLESS(VectorDrawableOp) 353 , vectorDrawable(tree) {} 354 VectorDrawable::Tree* vectorDrawable; 355 }; 356 357 /** 358 * Real-time, dynamic-lit shadow. 359 * 360 * Uses invalid/empty bounds and matrix since ShadowOp bounds aren't known at defer time, 361 * and are resolved dynamically, and transform isn't needed. 362 * 363 * State construction handles these properties specially, ignoring matrix/bounds. 364 */ 365 struct ShadowOp : RecordedOp { ShadowOpShadowOp366 ShadowOp(sp<TessellationCache::ShadowTask>& shadowTask, float casterAlpha) 367 : RecordedOp(RecordedOpId::ShadowOp, Rect(), Matrix4::identity(), nullptr, nullptr) 368 , shadowTask(shadowTask) 369 , casterAlpha(casterAlpha) { 370 }; 371 sp<TessellationCache::ShadowTask> shadowTask; 372 const float casterAlpha; 373 }; 374 375 struct SimpleRectsOp : RecordedOp { // Filled, no AA (TODO: better name?) SimpleRectsOpSimpleRectsOp376 SimpleRectsOp(BASE_PARAMS, Vertex* vertices, size_t vertexCount) 377 : SUPER(SimpleRectsOp) 378 , vertices(vertices) 379 , vertexCount(vertexCount) {} 380 Vertex* vertices; 381 const size_t vertexCount; 382 }; 383 384 struct TextOp : RecordedOp { TextOpTextOp385 TextOp(BASE_PARAMS, const glyph_t* glyphs, const float* positions, int glyphCount, 386 float x, float y) 387 : SUPER(TextOp) 388 , glyphs(glyphs) 389 , positions(positions) 390 , glyphCount(glyphCount) 391 , x(x) 392 , y(y) {} 393 const glyph_t* glyphs; 394 const float* positions; 395 const int glyphCount; 396 const float x; 397 const float y; 398 }; 399 400 struct TextOnPathOp : RecordedOp { 401 // TODO: explicitly define bounds TextOnPathOpTextOnPathOp402 TextOnPathOp(const Matrix4& localMatrix, const ClipBase* localClip, const SkPaint* paint, 403 const glyph_t* glyphs, int glyphCount, const SkPath* path, float hOffset, float vOffset) 404 : RecordedOp(RecordedOpId::TextOnPathOp, Rect(), localMatrix, localClip, paint) 405 , glyphs(glyphs) 406 , glyphCount(glyphCount) 407 , path(path) 408 , hOffset(hOffset) 409 , vOffset(vOffset) {} 410 const glyph_t* glyphs; 411 const int glyphCount; 412 413 const SkPath* path; 414 const float hOffset; 415 const float vOffset; 416 }; 417 418 struct TextureLayerOp : RecordedOp { TextureLayerOpTextureLayerOp419 TextureLayerOp(BASE_PARAMS_PAINTLESS, Layer* layer) 420 : SUPER_PAINTLESS(TextureLayerOp) 421 , layer(layer) {} 422 423 // Copy an existing TextureLayerOp, replacing the underlying matrix TextureLayerOpTextureLayerOp424 TextureLayerOp(const TextureLayerOp& op, const Matrix4& replacementMatrix) 425 : RecordedOp(RecordedOpId::TextureLayerOp, op.unmappedBounds, replacementMatrix, 426 op.localClip, op.paint) 427 , layer(op.layer) { 428 429 } 430 Layer* layer; 431 }; 432 433 //////////////////////////////////////////////////////////////////////////////////////////////////// 434 // Layers 435 //////////////////////////////////////////////////////////////////////////////////////////////////// 436 437 /** 438 * Stateful operation! denotes the creation of an off-screen layer, 439 * and that commands following will render into it. 440 */ 441 struct BeginLayerOp : RecordedOp { BeginLayerOpBeginLayerOp442 BeginLayerOp(BASE_PARAMS) 443 : SUPER(BeginLayerOp) {} 444 }; 445 446 /** 447 * Stateful operation! Denotes end of off-screen layer, and that 448 * commands since last BeginLayerOp should be drawn into parent FBO. 449 * 450 * State in this op is empty, it just serves to signal that a layer has been finished. 451 */ 452 struct EndLayerOp : RecordedOp { EndLayerOpEndLayerOp453 EndLayerOp() 454 : RecordedOp(RecordedOpId::EndLayerOp, Rect(), Matrix4::identity(), nullptr, nullptr) {} 455 }; 456 457 struct BeginUnclippedLayerOp : RecordedOp { BeginUnclippedLayerOpBeginUnclippedLayerOp458 BeginUnclippedLayerOp(BASE_PARAMS) 459 : SUPER(BeginUnclippedLayerOp) {} 460 }; 461 462 struct EndUnclippedLayerOp : RecordedOp { EndUnclippedLayerOpEndUnclippedLayerOp463 EndUnclippedLayerOp() 464 : RecordedOp(RecordedOpId::EndUnclippedLayerOp, Rect(), Matrix4::identity(), nullptr, nullptr) {} 465 }; 466 467 struct CopyToLayerOp : RecordedOp { CopyToLayerOpCopyToLayerOp468 CopyToLayerOp(const RecordedOp& op, OffscreenBuffer** layerHandle) 469 : RecordedOp(RecordedOpId::CopyToLayerOp, 470 op.unmappedBounds, 471 op.localMatrix, 472 nullptr, // clip intentionally ignored 473 op.paint) 474 , layerHandle(layerHandle) {} 475 476 // Records a handle to the Layer object, since the Layer itself won't be 477 // constructed until after this operation is constructed. 478 OffscreenBuffer** layerHandle; 479 }; 480 481 482 // draw the parameter layer underneath 483 struct CopyFromLayerOp : RecordedOp { CopyFromLayerOpCopyFromLayerOp484 CopyFromLayerOp(const RecordedOp& op, OffscreenBuffer** layerHandle) 485 : RecordedOp(RecordedOpId::CopyFromLayerOp, 486 op.unmappedBounds, 487 op.localMatrix, 488 nullptr, // clip intentionally ignored 489 op.paint) 490 , layerHandle(layerHandle) {} 491 492 // Records a handle to the Layer object, since the Layer itself won't be 493 // constructed until after this operation is constructed. 494 OffscreenBuffer** layerHandle; 495 }; 496 497 /** 498 * Draws an OffscreenBuffer. 499 * 500 * Alpha, mode, and colorfilter are embedded, since LayerOps are always dynamically generated, 501 * when creating/tracking a SkPaint* during defer isn't worth the bother. 502 */ 503 struct LayerOp : RecordedOp { 504 // Records a one-use (saveLayer) layer for drawing. LayerOpLayerOp505 LayerOp(BASE_PARAMS, OffscreenBuffer** layerHandle) 506 : SUPER_PAINTLESS(LayerOp) 507 , layerHandle(layerHandle) 508 , alpha(paint ? paint->getAlpha() / 255.0f : 1.0f) 509 , mode(PaintUtils::getXfermodeDirect(paint)) 510 , colorFilter(paint ? paint->getColorFilter() : nullptr) {} 511 LayerOpLayerOp512 LayerOp(RenderNode& node) 513 : RecordedOp(RecordedOpId::LayerOp, Rect(node.getWidth(), node.getHeight()), Matrix4::identity(), nullptr, nullptr) 514 , layerHandle(node.getLayerHandle()) 515 , alpha(node.properties().layerProperties().alpha() / 255.0f) 516 , mode(node.properties().layerProperties().xferMode()) 517 , colorFilter(node.properties().layerProperties().colorFilter()) {} 518 519 // Records a handle to the Layer object, since the Layer itself won't be 520 // constructed until after this operation is constructed. 521 OffscreenBuffer** layerHandle; 522 const float alpha; 523 const SkXfermode::Mode mode; 524 525 // pointer to object owned by either LayerProperties, or a recorded Paint object in a 526 // BeginLayerOp. Lives longer than LayerOp in either case, so no skia ref counting is used. 527 SkColorFilter* colorFilter; 528 }; 529 530 }; // namespace uirenderer 531 }; // namespace android 532 533 #endif // ANDROID_HWUI_RECORDED_OP_H 534