blob: 8c3a272bb389b4a889c72d12a4a448d304311e73 [file] [log] [blame]
ztenghui55bfb4e2013-12-03 10:38:55 -08001/*
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
ztenghuid5e8ade2014-08-13 15:48:02 -070020#include "utils/MathUtils.h"
ztenghui55bfb4e2013-12-03 10:38:55 -080021
22namespace android {
23namespace uirenderer {
24
25class VertexBuffer {
26public:
Chris Craik05f3d6e2014-06-02 16:27:04 -070027 enum Mode {
28 kStandard = 0,
29 kOnePolyRingShadow = 1,
ztenghuid5e8ade2014-08-13 15:48:02 -070030 kTwoPolyRingShadow = 2,
31 kIndices = 3
Chris Craik05f3d6e2014-06-02 16:27:04 -070032 };
33
34 VertexBuffer()
35 : mBuffer(0)
ztenghuid5e8ade2014-08-13 15:48:02 -070036 , mIndices(0)
Chris Craik05f3d6e2014-06-02 16:27:04 -070037 , mVertexCount(0)
ztenghuid5e8ade2014-08-13 15:48:02 -070038 , mIndexCount(0)
39 , mAllocatedVertexCount(0)
40 , mAllocatedIndexCount(0)
Chris Craik05f3d6e2014-06-02 16:27:04 -070041 , mByteCount(0)
42 , mMode(kStandard)
ztenghuid5e8ade2014-08-13 15:48:02 -070043 , mReallocBuffer(0)
Chris Craik05f3d6e2014-06-02 16:27:04 -070044 , mCleanupMethod(NULL)
ztenghuid5e8ade2014-08-13 15:48:02 -070045 , mCleanupIndexMethod(NULL)
ztenghui55bfb4e2013-12-03 10:38:55 -080046 {}
47
48 ~VertexBuffer() {
49 if (mCleanupMethod) mCleanupMethod(mBuffer);
ztenghuid5e8ade2014-08-13 15:48:02 -070050 if (mCleanupIndexMethod) mCleanupIndexMethod(mIndices);
ztenghui55bfb4e2013-12-03 10:38:55 -080051 }
52
53 /**
54 This should be the only method used by the Tessellator. Subsequent calls to
55 alloc will allocate space within the first allocation (useful if you want to
56 eventually allocate multiple regions within a single VertexBuffer, such as
Chris Craik05f3d6e2014-06-02 16:27:04 -070057 with PathTessellator::tessellateLines())
ztenghui55bfb4e2013-12-03 10:38:55 -080058 */
59 template <class TYPE>
60 TYPE* alloc(int vertexCount) {
61 if (mVertexCount) {
62 TYPE* reallocBuffer = (TYPE*)mReallocBuffer;
63 // already have allocated the buffer, re-allocate space within
64 if (mReallocBuffer != mBuffer) {
65 // not first re-allocation, leave space for degenerate triangles to separate strips
66 reallocBuffer += 2;
67 }
68 mReallocBuffer = reallocBuffer + vertexCount;
69 return reallocBuffer;
70 }
ztenghuid5e8ade2014-08-13 15:48:02 -070071 mAllocatedVertexCount = vertexCount;
ztenghui55bfb4e2013-12-03 10:38:55 -080072 mVertexCount = vertexCount;
Chris Craik05f3d6e2014-06-02 16:27:04 -070073 mByteCount = mVertexCount * sizeof(TYPE);
ztenghui55bfb4e2013-12-03 10:38:55 -080074 mReallocBuffer = mBuffer = (void*)new TYPE[vertexCount];
Chris Craikc93e45c2014-07-16 10:15:56 -070075
ztenghui55bfb4e2013-12-03 10:38:55 -080076 mCleanupMethod = &(cleanup<TYPE>);
77
78 return (TYPE*)mBuffer;
79 }
80
81 template <class TYPE>
ztenghuid5e8ade2014-08-13 15:48:02 -070082 TYPE* allocIndices(int indexCount) {
83 mAllocatedIndexCount = indexCount;
84 mIndexCount = indexCount;
85 mIndices = (void*)new TYPE[indexCount];
86
87 mCleanupIndexMethod = &(cleanup<TYPE>);
88
89 return (TYPE*)mIndices;
90 }
91
92 template <class TYPE>
ztenghui55bfb4e2013-12-03 10:38:55 -080093 void copyInto(const VertexBuffer& srcBuffer, float xOffset, float yOffset) {
94 int verticesToCopy = srcBuffer.getVertexCount();
95
96 TYPE* dst = alloc<TYPE>(verticesToCopy);
97 TYPE* src = (TYPE*)srcBuffer.getBuffer();
98
99 for (int i = 0; i < verticesToCopy; i++) {
100 TYPE::copyWithOffset(&dst[i], src[i], xOffset, yOffset);
101 }
102 }
103
Chris Craikc93e45c2014-07-16 10:15:56 -0700104 /**
105 * Brute force bounds computation, used only if the producer of this
106 * vertex buffer can't determine bounds more simply/efficiently
107 */
108 template <class TYPE>
Chris Craik9a89bc62014-07-23 17:21:25 -0700109 void computeBounds(int vertexCount = 0) {
Chris Craikc93e45c2014-07-16 10:15:56 -0700110 if (!mVertexCount) {
111 mBounds.setEmpty();
112 return;
113 }
Chris Craik9a89bc62014-07-23 17:21:25 -0700114
115 // default: compute over every vertex
116 if (vertexCount == 0) vertexCount = mVertexCount;
117
Chris Craikc93e45c2014-07-16 10:15:56 -0700118 TYPE* current = (TYPE*)mBuffer;
Chris Craik9a89bc62014-07-23 17:21:25 -0700119 TYPE* end = current + vertexCount;
Chris Craikc93e45c2014-07-16 10:15:56 -0700120 mBounds.set(current->x, current->y, current->x, current->y);
121 for (; current < end; current++) {
122 mBounds.expandToCoverVertex(current->x, current->y);
123 }
124 }
125
ztenghui55bfb4e2013-12-03 10:38:55 -0800126 const void* getBuffer() const { return mBuffer; }
ztenghuid5e8ade2014-08-13 15:48:02 -0700127 const void* getIndices() const { return mIndices; }
Chris Craik05f3d6e2014-06-02 16:27:04 -0700128 const Rect& getBounds() const { return mBounds; }
ztenghui55bfb4e2013-12-03 10:38:55 -0800129 unsigned int getVertexCount() const { return mVertexCount; }
Chris Craik05f3d6e2014-06-02 16:27:04 -0700130 unsigned int getSize() const { return mByteCount; }
ztenghuid5e8ade2014-08-13 15:48:02 -0700131 unsigned int getIndexCount() const { return mIndexCount; }
132 void updateIndexCount(unsigned int newCount) {
133 mIndexCount = MathUtils::min(newCount, mAllocatedIndexCount);
134 }
135 void updateVertexCount(unsigned int newCount) {
ztenghui512e6432014-09-10 13:08:20 -0700136 mVertexCount = MathUtils::min(newCount, mAllocatedVertexCount);
ztenghuid5e8ade2014-08-13 15:48:02 -0700137 }
Chris Craik05f3d6e2014-06-02 16:27:04 -0700138 Mode getMode() const { return mMode; }
139
140 void setBounds(Rect bounds) { mBounds = bounds; }
141 void setMode(Mode mode) { mMode = mode; }
ztenghui55bfb4e2013-12-03 10:38:55 -0800142
143 template <class TYPE>
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
152private:
153 template <class TYPE>
154 static void cleanup(void* buffer) {
155 delete[] (TYPE*)buffer;
156 }
157
Chris Craik05f3d6e2014-06-02 16:27:04 -0700158 Rect mBounds;
ztenghuid5e8ade2014-08-13 15:48:02 -0700159
ztenghui55bfb4e2013-12-03 10:38:55 -0800160 void* mBuffer;
ztenghuid5e8ade2014-08-13 15:48:02 -0700161 void* mIndices;
162
ztenghui55bfb4e2013-12-03 10:38:55 -0800163 unsigned int mVertexCount;
ztenghuid5e8ade2014-08-13 15:48:02 -0700164 unsigned int mIndexCount;
165 unsigned int mAllocatedVertexCount;
166 unsigned int mAllocatedIndexCount;
Chris Craik05f3d6e2014-06-02 16:27:04 -0700167 unsigned int mByteCount;
ztenghuid5e8ade2014-08-13 15:48:02 -0700168
Chris Craik05f3d6e2014-06-02 16:27:04 -0700169 Mode mMode;
ztenghui55bfb4e2013-12-03 10:38:55 -0800170
171 void* mReallocBuffer; // used for multi-allocation
172
173 void (*mCleanupMethod)(void*);
ztenghuid5e8ade2014-08-13 15:48:02 -0700174 void (*mCleanupIndexMethod)(void*);
ztenghui55bfb4e2013-12-03 10:38:55 -0800175};
176
177}; // namespace uirenderer
178}; // namespace android
179
180#endif // ANDROID_HWUI_VERTEX_BUFFER_H