1 /*
2  * Copyright (C) 2012 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_VERTEX_BUFFER_H
18 #define ANDROID_HWUI_VERTEX_BUFFER_H
19 
20 #include <algorithm>
21 
22 namespace android {
23 namespace uirenderer {
24 
25 class VertexBuffer {
26 public:
27     enum MeshFeatureFlags {
28         kNone = 0,
29         kAlpha = 1 << 0,
30         kIndices = 1 << 1,
31     };
32 
VertexBuffer()33     VertexBuffer()
34             : mBuffer(nullptr)
35             , mIndices(nullptr)
36             , mVertexCount(0)
37             , mIndexCount(0)
38             , mAllocatedVertexCount(0)
39             , mAllocatedIndexCount(0)
40             , mByteCount(0)
41             , mMeshFeatureFlags(kNone)
42             , mReallocBuffer(nullptr)
43             , mCleanupMethod(nullptr)
44             , mCleanupIndexMethod(nullptr) {}
45 
~VertexBuffer()46     ~VertexBuffer() {
47         if (mCleanupMethod) mCleanupMethod(mBuffer);
48         if (mCleanupIndexMethod) mCleanupIndexMethod(mIndices);
49     }
50 
51     /**
52        This should be the only method used by the Tessellator. Subsequent calls to
53        alloc will allocate space within the first allocation (useful if you want to
54        eventually allocate multiple regions within a single VertexBuffer, such as
55        with PathTessellator::tessellateLines())
56      */
57     template <class TYPE>
alloc(int vertexCount)58     TYPE* alloc(int vertexCount) {
59         if (mVertexCount) {
60             TYPE* reallocBuffer = (TYPE*)mReallocBuffer;
61             // already have allocated the buffer, re-allocate space within
62             if (mReallocBuffer != mBuffer) {
63                 // not first re-allocation, leave space for degenerate triangles to separate strips
64                 reallocBuffer += 2;
65             }
66             mReallocBuffer = reallocBuffer + vertexCount;
67             return reallocBuffer;
68         }
69         mAllocatedVertexCount = vertexCount;
70         mVertexCount = vertexCount;
71         mByteCount = mVertexCount * sizeof(TYPE);
72         mReallocBuffer = mBuffer = (void*)new TYPE[vertexCount];
73 
74         mCleanupMethod = &(cleanup<TYPE>);
75 
76         return (TYPE*)mBuffer;
77     }
78 
79     template <class TYPE>
allocIndices(int indexCount)80     TYPE* allocIndices(int indexCount) {
81         mAllocatedIndexCount = indexCount;
82         mIndexCount = indexCount;
83         mIndices = (void*)new TYPE[indexCount];
84 
85         mCleanupIndexMethod = &(cleanup<TYPE>);
86 
87         return (TYPE*)mIndices;
88     }
89 
90     template <class TYPE>
copyInto(const VertexBuffer & srcBuffer,float xOffset,float yOffset)91     void copyInto(const VertexBuffer& srcBuffer, float xOffset, float yOffset) {
92         int verticesToCopy = srcBuffer.getVertexCount();
93 
94         TYPE* dst = alloc<TYPE>(verticesToCopy);
95         TYPE* src = (TYPE*)srcBuffer.getBuffer();
96 
97         for (int i = 0; i < verticesToCopy; i++) {
98             TYPE::copyWithOffset(&dst[i], src[i], xOffset, yOffset);
99         }
100     }
101 
102     /**
103      * Brute force bounds computation, used only if the producer of this
104      * vertex buffer can't determine bounds more simply/efficiently
105      */
106     template <class TYPE>
107     void computeBounds(int vertexCount = 0) {
108         if (!mVertexCount) {
109             mBounds.setEmpty();
110             return;
111         }
112 
113         // default: compute over every vertex
114         if (vertexCount == 0) vertexCount = mVertexCount;
115 
116         TYPE* current = (TYPE*)mBuffer;
117         TYPE* end = current + vertexCount;
118         mBounds.set(current->x, current->y, current->x, current->y);
119         for (; current < end; current++) {
120             mBounds.expandToCover(current->x, current->y);
121         }
122     }
123 
getBuffer()124     const void* getBuffer() const { return mBuffer; }
getIndices()125     const void* getIndices() const { return mIndices; }
getBounds()126     const Rect& getBounds() const { return mBounds; }
getVertexCount()127     unsigned int getVertexCount() const { return mVertexCount; }
getSize()128     unsigned int getSize() const { return mByteCount; }
getIndexCount()129     unsigned int getIndexCount() const { return mIndexCount; }
updateIndexCount(unsigned int newCount)130     void updateIndexCount(unsigned int newCount) {
131         mIndexCount = std::min(newCount, mAllocatedIndexCount);
132     }
updateVertexCount(unsigned int newCount)133     void updateVertexCount(unsigned int newCount) {
134         mVertexCount = std::min(newCount, mAllocatedVertexCount);
135     }
getMeshFeatureFlags()136     MeshFeatureFlags getMeshFeatureFlags() const { return mMeshFeatureFlags; }
setMeshFeatureFlags(int flags)137     void setMeshFeatureFlags(int flags) {
138         mMeshFeatureFlags = static_cast<MeshFeatureFlags>(flags);
139     }
140 
setBounds(Rect bounds)141     void setBounds(Rect bounds) { mBounds = bounds; }
142 
143     template <class TYPE>
createDegenerateSeparators(int allocSize)144     void createDegenerateSeparators(int allocSize) {
145         TYPE* end = (TYPE*)mBuffer + mVertexCount;
146         for (TYPE* degen = (TYPE*)mBuffer + allocSize; degen < end; degen += 2 + allocSize) {
147             memcpy(degen, degen - 1, sizeof(TYPE));
148             memcpy(degen + 1, degen + 2, sizeof(TYPE));
149         }
150     }
151 
152 private:
153     template <class TYPE>
cleanup(void * buffer)154     static void cleanup(void* buffer) {
155         delete[](TYPE*) buffer;
156     }
157 
158     Rect mBounds;
159 
160     void* mBuffer;
161     void* mIndices;
162 
163     unsigned int mVertexCount;
164     unsigned int mIndexCount;
165     unsigned int mAllocatedVertexCount;
166     unsigned int mAllocatedIndexCount;
167     unsigned int mByteCount;
168 
169     MeshFeatureFlags mMeshFeatureFlags;
170 
171     void* mReallocBuffer;  // used for multi-allocation
172 
173     void (*mCleanupMethod)(void*);
174     void (*mCleanupIndexMethod)(void*);
175 };
176 
177 }  // namespace uirenderer
178 }  // namespace android
179 
180 #endif  // ANDROID_HWUI_VERTEX_BUFFER_H
181