1 /*
2  * Copyright (C) 2013 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_TESSELLATION_CACHE_H
18 #define ANDROID_HWUI_TESSELLATION_CACHE_H
19 
20 #include <utils/LruCache.h>
21 #include <utils/Mutex.h>
22 #include <utils/Vector.h>
23 
24 #include "Debug.h"
25 #include "utils/Macros.h"
26 #include "utils/Pair.h"
27 #include "VertexBuffer.h"
28 
29 class SkBitmap;
30 class SkCanvas;
31 class SkPaint;
32 class SkPath;
33 struct SkRect;
34 
35 namespace android {
36 namespace uirenderer {
37 
38 class Caches;
39 
40 ///////////////////////////////////////////////////////////////////////////////
41 // Classes
42 ///////////////////////////////////////////////////////////////////////////////
43 
44 class TessellationCache {
45 public:
46     typedef Pair<VertexBuffer*, VertexBuffer*> vertexBuffer_pair_t;
47 
48     struct Description {
49         DESCRIPTION_TYPE(Description);
50         enum Type {
51             kNone,
52             kRoundRect,
53         };
54 
55         Type type;
56         float scaleX;
57         float scaleY;
58         bool aa;
59         SkPaint::Cap cap;
60         SkPaint::Style style;
61         float strokeWidth;
62         union Shape {
63             struct RoundRect {
64                 float width;
65                 float height;
66                 float rx;
67                 float ry;
68             } roundRect;
69         } shape;
70 
71         Description();
72         Description(Type type, const Matrix4& transform, const SkPaint& paint);
73         hash_t hash() const;
74         void setupMatrixAndPaint(Matrix4* matrix, SkPaint* paint) const;
75     };
76 
77     struct ShadowDescription {
78         DESCRIPTION_TYPE(ShadowDescription);
79         const void* nodeKey;
80         float matrixData[16];
81 
82         ShadowDescription();
83         ShadowDescription(const void* nodeKey, const Matrix4* drawTransform);
84         hash_t hash() const;
85     };
86 
87     TessellationCache();
88     ~TessellationCache();
89 
90     /**
91      * Clears the cache. This causes all TessellationBuffers to be deleted.
92      */
93     void clear();
94 
95     /**
96      * Sets the maximum size of the cache in bytes.
97      */
98     void setMaxSize(uint32_t maxSize);
99     /**
100      * Returns the maximum size of the cache in bytes.
101      */
102     uint32_t getMaxSize();
103     /**
104      * Returns the current size of the cache in bytes.
105      */
106     uint32_t getSize();
107 
108     /**
109      * Trims the contents of the cache, removing items until it's under its
110      * specified limit.
111      *
112      * Trimming is used for caches that support pre-caching from a worker
113      * thread. During pre-caching the maximum limit of the cache can be
114      * exceeded for the duration of the frame. It is therefore required to
115      * trim the cache at the end of the frame to keep the total amount of
116      * memory used under control.
117      *
118      * Also removes transient Shadow VertexBuffers, which aren't cached between frames.
119      */
120     void trim();
121 
122     // TODO: precache/get for Oval, Lines, Points, etc.
123 
precacheRoundRect(const Matrix4 & transform,const SkPaint & paint,float width,float height,float rx,float ry)124     void precacheRoundRect(const Matrix4& transform, const SkPaint& paint,
125             float width, float height, float rx, float ry) {
126         getRoundRectBuffer(transform, paint, width, height, rx, ry);
127     }
128     const VertexBuffer* getRoundRect(const Matrix4& transform, const SkPaint& paint,
129             float width, float height, float rx, float ry);
130 
131     void precacheShadows(const Matrix4* drawTransform, const Rect& localClip,
132             bool opaque, const SkPath* casterPerimeter,
133             const Matrix4* transformXY, const Matrix4* transformZ,
134             const Vector3& lightCenter, float lightRadius);
135 
136     void getShadowBuffers(const Matrix4* drawTransform, const Rect& localClip,
137             bool opaque, const SkPath* casterPerimeter,
138             const Matrix4* transformXY, const Matrix4* transformZ,
139             const Vector3& lightCenter, float lightRadius,
140             vertexBuffer_pair_t& outBuffers);
141 
142 private:
143     class Buffer;
144     class TessellationTask;
145     class TessellationProcessor;
146 
147     typedef VertexBuffer* (*Tessellator)(const Description&);
148 
149     Buffer* getRectBuffer(const Matrix4& transform, const SkPaint& paint,
150             float width, float height);
151     Buffer* getRoundRectBuffer(const Matrix4& transform, const SkPaint& paint,
152             float width, float height, float rx, float ry);
153 
154     Buffer* getOrCreateBuffer(const Description& entry, Tessellator tessellator);
155 
156     uint32_t mSize;
157     uint32_t mMaxSize;
158 
159     bool mDebugEnabled;
160 
161     mutable Mutex mLock;
162 
163     ///////////////////////////////////////////////////////////////////////////////
164     // General tessellation caching
165     ///////////////////////////////////////////////////////////////////////////////
166     sp<TaskProcessor<VertexBuffer*> > mProcessor;
167     LruCache<Description, Buffer*> mCache;
168     class BufferRemovedListener : public OnEntryRemoved<Description, Buffer*> {
169         void operator()(Description& description, Buffer*& buffer);
170     };
171     BufferRemovedListener mBufferRemovedListener;
172 
173     ///////////////////////////////////////////////////////////////////////////////
174     // Shadow tessellation caching
175     ///////////////////////////////////////////////////////////////////////////////
176     sp<TaskProcessor<vertexBuffer_pair_t*> > mShadowProcessor;
177 
178     // holds a pointer, and implicit strong ref to each shadow task of the frame
179     LruCache<ShadowDescription, Task<vertexBuffer_pair_t*>*> mShadowCache;
180     class BufferPairRemovedListener : public OnEntryRemoved<ShadowDescription, Task<vertexBuffer_pair_t*>*> {
operator()181         void operator()(ShadowDescription& description, Task<vertexBuffer_pair_t*>*& bufferPairTask) {
182             bufferPairTask->decStrong(NULL);
183         }
184     };
185     BufferPairRemovedListener mBufferPairRemovedListener;
186 
187 }; // class TessellationCache
188 
189 }; // namespace uirenderer
190 }; // namespace android
191 
192 #endif // ANDROID_HWUI_PATH_CACHE_H
193