1 /*
2 * Copyright 2013 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "src/gpu/GrCpuBuffer.h"
9 #include "src/gpu/gl/GrGLBuffer.h"
10 #include "src/gpu/gl/GrGLGpu.h"
11 #include "src/gpu/gl/GrGLVertexArray.h"
12
13 struct AttribLayout {
14 bool fNormalized; // Only used by floating point types.
15 uint8_t fCount;
16 uint16_t fType;
17 };
18
19 static_assert(4 == sizeof(AttribLayout));
20
attrib_layout(GrVertexAttribType type)21 static AttribLayout attrib_layout(GrVertexAttribType type) {
22 switch (type) {
23 case kFloat_GrVertexAttribType:
24 return {false, 1, GR_GL_FLOAT};
25 case kFloat2_GrVertexAttribType:
26 return {false, 2, GR_GL_FLOAT};
27 case kFloat3_GrVertexAttribType:
28 return {false, 3, GR_GL_FLOAT};
29 case kFloat4_GrVertexAttribType:
30 return {false, 4, GR_GL_FLOAT};
31 case kHalf_GrVertexAttribType:
32 return {false, 1, GR_GL_HALF_FLOAT};
33 case kHalf2_GrVertexAttribType:
34 return {false, 2, GR_GL_HALF_FLOAT};
35 case kHalf4_GrVertexAttribType:
36 return {false, 4, GR_GL_HALF_FLOAT};
37 case kInt2_GrVertexAttribType:
38 return {false, 2, GR_GL_INT};
39 case kInt3_GrVertexAttribType:
40 return {false, 3, GR_GL_INT};
41 case kInt4_GrVertexAttribType:
42 return {false, 4, GR_GL_INT};
43 case kByte_GrVertexAttribType:
44 return {false, 1, GR_GL_BYTE};
45 case kByte2_GrVertexAttribType:
46 return {false, 2, GR_GL_BYTE};
47 case kByte4_GrVertexAttribType:
48 return {false, 4, GR_GL_BYTE};
49 case kUByte_GrVertexAttribType:
50 return {false, 1, GR_GL_UNSIGNED_BYTE};
51 case kUByte2_GrVertexAttribType:
52 return {false, 2, GR_GL_UNSIGNED_BYTE};
53 case kUByte4_GrVertexAttribType:
54 return {false, 4, GR_GL_UNSIGNED_BYTE};
55 case kUByte_norm_GrVertexAttribType:
56 return {true, 1, GR_GL_UNSIGNED_BYTE};
57 case kUByte4_norm_GrVertexAttribType:
58 return {true, 4, GR_GL_UNSIGNED_BYTE};
59 case kShort2_GrVertexAttribType:
60 return {false, 2, GR_GL_SHORT};
61 case kShort4_GrVertexAttribType:
62 return {false, 4, GR_GL_SHORT};
63 case kUShort2_GrVertexAttribType:
64 return {false, 2, GR_GL_UNSIGNED_SHORT};
65 case kUShort2_norm_GrVertexAttribType:
66 return {true, 2, GR_GL_UNSIGNED_SHORT};
67 case kInt_GrVertexAttribType:
68 return {false, 1, GR_GL_INT};
69 case kUInt_GrVertexAttribType:
70 return {false, 1, GR_GL_UNSIGNED_INT};
71 case kUShort_norm_GrVertexAttribType:
72 return {true, 1, GR_GL_UNSIGNED_SHORT};
73 case kUShort4_norm_GrVertexAttribType:
74 return {true, 4, GR_GL_UNSIGNED_SHORT};
75 }
76 SK_ABORT("Unknown vertex attrib type");
77 };
78
set(GrGLGpu * gpu,int index,const GrBuffer * vertexBuffer,GrVertexAttribType cpuType,GrSLType gpuType,GrGLsizei stride,size_t offsetInBytes,int divisor)79 void GrGLAttribArrayState::set(GrGLGpu* gpu,
80 int index,
81 const GrBuffer* vertexBuffer,
82 GrVertexAttribType cpuType,
83 GrSLType gpuType,
84 GrGLsizei stride,
85 size_t offsetInBytes,
86 int divisor) {
87 SkASSERT(index >= 0 && index < fAttribArrayStates.count());
88 SkASSERT(0 == divisor || gpu->caps()->drawInstancedSupport());
89 AttribArrayState* array = &fAttribArrayStates[index];
90 const char* offsetAsPtr;
91 bool bufferChanged = false;
92 if (vertexBuffer->isCpuBuffer()) {
93 if (!array->fUsingCpuBuffer) {
94 bufferChanged = true;
95 array->fUsingCpuBuffer = true;
96 }
97 offsetAsPtr = static_cast<const GrCpuBuffer*>(vertexBuffer)->data() + offsetInBytes;
98 } else {
99 auto gpuBuffer = static_cast<const GrGpuBuffer*>(vertexBuffer);
100 if (array->fUsingCpuBuffer || array->fVertexBufferUniqueID != gpuBuffer->uniqueID()) {
101 bufferChanged = true;
102 array->fVertexBufferUniqueID = gpuBuffer->uniqueID();
103 }
104 offsetAsPtr = reinterpret_cast<const char*>(offsetInBytes);
105 }
106 if (bufferChanged ||
107 array->fCPUType != cpuType ||
108 array->fGPUType != gpuType ||
109 array->fStride != stride ||
110 array->fOffset != offsetAsPtr) {
111 // We always have to call this if we're going to change the array pointer. 'array' is
112 // tracking the last buffer used to setup attrib pointers, not the last buffer bound.
113 // GrGLGpu will avoid redundant binds.
114 gpu->bindBuffer(GrGpuBufferType::kVertex, vertexBuffer);
115 const AttribLayout& layout = attrib_layout(cpuType);
116 if (GrSLTypeIsFloatType(gpuType)) {
117 GR_GL_CALL(gpu->glInterface(), VertexAttribPointer(index,
118 layout.fCount,
119 layout.fType,
120 layout.fNormalized,
121 stride,
122 offsetAsPtr));
123 } else {
124 SkASSERT(gpu->caps()->shaderCaps()->integerSupport());
125 SkASSERT(!layout.fNormalized);
126 GR_GL_CALL(gpu->glInterface(), VertexAttribIPointer(index,
127 layout.fCount,
128 layout.fType,
129 stride,
130 offsetAsPtr));
131 }
132 array->fCPUType = cpuType;
133 array->fGPUType = gpuType;
134 array->fStride = stride;
135 array->fOffset = offsetAsPtr;
136 }
137 if (gpu->caps()->drawInstancedSupport() && array->fDivisor != divisor) {
138 SkASSERT(0 == divisor || 1 == divisor); // not necessarily a requirement but what we expect.
139 GR_GL_CALL(gpu->glInterface(), VertexAttribDivisor(index, divisor));
140 array->fDivisor = divisor;
141 }
142 }
143
enableVertexArrays(const GrGLGpu * gpu,int enabledCount,GrPrimitiveRestart enablePrimitiveRestart)144 void GrGLAttribArrayState::enableVertexArrays(const GrGLGpu* gpu, int enabledCount,
145 GrPrimitiveRestart enablePrimitiveRestart) {
146 SkASSERT(enabledCount <= fAttribArrayStates.count());
147
148 if (!fEnableStateIsValid || enabledCount != fNumEnabledArrays) {
149 int firstIdxToEnable = fEnableStateIsValid ? fNumEnabledArrays : 0;
150 for (int i = firstIdxToEnable; i < enabledCount; ++i) {
151 GR_GL_CALL(gpu->glInterface(), EnableVertexAttribArray(i));
152 }
153
154 int endIdxToDisable = fEnableStateIsValid ? fNumEnabledArrays : fAttribArrayStates.count();
155 for (int i = enabledCount; i < endIdxToDisable; ++i) {
156 GR_GL_CALL(gpu->glInterface(), DisableVertexAttribArray(i));
157 }
158
159 fNumEnabledArrays = enabledCount;
160 }
161
162 SkASSERT(GrPrimitiveRestart::kNo == enablePrimitiveRestart ||
163 gpu->caps()->usePrimitiveRestart());
164
165 if (gpu->caps()->usePrimitiveRestart() &&
166 (!fEnableStateIsValid || enablePrimitiveRestart != fPrimitiveRestartEnabled)) {
167 if (GrPrimitiveRestart::kYes == enablePrimitiveRestart) {
168 GR_GL_CALL(gpu->glInterface(), Enable(GR_GL_PRIMITIVE_RESTART_FIXED_INDEX));
169 } else {
170 GR_GL_CALL(gpu->glInterface(), Disable(GR_GL_PRIMITIVE_RESTART_FIXED_INDEX));
171 }
172
173 fPrimitiveRestartEnabled = enablePrimitiveRestart;
174 }
175
176 fEnableStateIsValid = true;
177 }
178
179 ///////////////////////////////////////////////////////////////////////////////////////////////////
180
GrGLVertexArray(GrGLint id,int attribCount)181 GrGLVertexArray::GrGLVertexArray(GrGLint id, int attribCount)
182 : fID(id)
183 , fAttribArrays(attribCount)
184 , fIndexBufferUniqueID(SK_InvalidUniqueID) {
185 }
186
bind(GrGLGpu * gpu)187 GrGLAttribArrayState* GrGLVertexArray::bind(GrGLGpu* gpu) {
188 if (0 == fID) {
189 return nullptr;
190 }
191 gpu->bindVertexArray(fID);
192 return &fAttribArrays;
193 }
194
bindWithIndexBuffer(GrGLGpu * gpu,const GrBuffer * ibuff)195 GrGLAttribArrayState* GrGLVertexArray::bindWithIndexBuffer(GrGLGpu* gpu, const GrBuffer* ibuff) {
196 GrGLAttribArrayState* state = this->bind(gpu);
197 if (!state) {
198 return nullptr;
199 }
200 if (ibuff->isCpuBuffer()) {
201 GR_GL_CALL(gpu->glInterface(), BindBuffer(GR_GL_ELEMENT_ARRAY_BUFFER, 0));
202 } else {
203 const GrGLBuffer* glBuffer = static_cast<const GrGLBuffer*>(ibuff);
204 if (fIndexBufferUniqueID != glBuffer->uniqueID()) {
205 GR_GL_CALL(gpu->glInterface(),
206 BindBuffer(GR_GL_ELEMENT_ARRAY_BUFFER, glBuffer->bufferID()));
207 fIndexBufferUniqueID = glBuffer->uniqueID();
208 }
209 }
210 return state;
211 }
212
invalidateCachedState()213 void GrGLVertexArray::invalidateCachedState() {
214 fAttribArrays.invalidate();
215 fIndexBufferUniqueID.makeInvalid();
216 }
217