blob: 76fe62da88308b7d9e4ea6cf09738ced44dd4d7a [file] [log] [blame]
Jason Sams7c878f32009-06-30 14:13:04 -07001/*
2 * Copyright (C) 2009 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#include "rsContext.h"
Alex Sakhartchoukd0f5bd12011-01-31 14:53:24 -080018#ifndef ANDROID_RS_SERIALIZE
Alex Sakhartchoukaa7d2882010-05-21 12:53:13 -070019#include <GLES/gl.h>
20#include <GLES2/gl2.h>
21#include <GLES/glext.h>
Alex Sakhartchoukaa7d2882010-05-21 12:53:13 -070022#endif
23
Jason Sams7c878f32009-06-30 14:13:04 -070024using namespace android;
25using namespace android::renderscript;
26
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080027Mesh::Mesh(Context *rsc) : ObjectBase(rsc) {
Jason Sams7c878f32009-06-30 14:13:04 -070028 mPrimitives = NULL;
Jason Sams64676f32009-07-08 18:01:53 -070029 mPrimitivesCount = 0;
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -070030 mVertexBuffers = NULL;
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -070031 mVertexBufferCount = 0;
Alex Sakhartchouka3b59602011-01-28 09:31:47 -080032
Alex Sakhartchoukd0f5bd12011-01-31 14:53:24 -080033#ifndef ANDROID_RS_SERIALIZE
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -080034 mAttribs = NULL;
35 mAttribAllocationIndex = NULL;
36
37 mAttribCount = 0;
Alex Sakhartchouka3b59602011-01-28 09:31:47 -080038#endif
Jason Sams7c878f32009-06-30 14:13:04 -070039}
40
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080041Mesh::~Mesh() {
42 if (mVertexBuffers) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -070043 delete[] mVertexBuffers;
44 }
45
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080046 if (mPrimitives) {
47 for (uint32_t i = 0; i < mPrimitivesCount; i ++) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -070048 delete mPrimitives[i];
49 }
50 delete[] mPrimitives;
51 }
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -080052
Alex Sakhartchoukd0f5bd12011-01-31 14:53:24 -080053#ifndef ANDROID_RS_SERIALIZE
Alex Sakhartchouked9f2102010-11-09 17:00:54 -080054 if (mAttribs) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -080055 delete[] mAttribs;
56 delete[] mAttribAllocationIndex;
57 }
Alex Sakhartchouka3b59602011-01-28 09:31:47 -080058#endif
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -080059}
60
Alex Sakhartchouka3b59602011-01-28 09:31:47 -080061void Mesh::serialize(OStream *stream) const {
62 // Need to identify ourselves
63 stream->addU32((uint32_t)getClassId());
64
65 String8 name(getName());
66 stream->addString(&name);
67
68 // Store number of vertex streams
69 stream->addU32(mVertexBufferCount);
70 for (uint32_t vCount = 0; vCount < mVertexBufferCount; vCount ++) {
71 mVertexBuffers[vCount]->serialize(stream);
72 }
73
74 stream->addU32(mPrimitivesCount);
75 // Store the primitives
76 for (uint32_t pCount = 0; pCount < mPrimitivesCount; pCount ++) {
77 Primitive_t * prim = mPrimitives[pCount];
78
79 stream->addU8((uint8_t)prim->mPrimitive);
80
81 if (prim->mIndexBuffer.get()) {
82 stream->addU32(1);
83 prim->mIndexBuffer->serialize(stream);
84 } else {
85 stream->addU32(0);
86 }
87 }
88}
89
90Mesh *Mesh::createFromStream(Context *rsc, IStream *stream) {
91 // First make sure we are reading the correct object
92 RsA3DClassID classID = (RsA3DClassID)stream->loadU32();
93 if (classID != RS_A3D_CLASS_ID_MESH) {
94 LOGE("mesh loading skipped due to invalid class id");
95 return NULL;
96 }
97
98 Mesh * mesh = new Mesh(rsc);
99
100 String8 name;
101 stream->loadString(&name);
102 mesh->setName(name.string(), name.size());
103
104 mesh->mVertexBufferCount = stream->loadU32();
105 if (mesh->mVertexBufferCount) {
106 mesh->mVertexBuffers = new ObjectBaseRef<Allocation>[mesh->mVertexBufferCount];
107
108 for (uint32_t vCount = 0; vCount < mesh->mVertexBufferCount; vCount ++) {
109 Allocation *vertexAlloc = Allocation::createFromStream(rsc, stream);
110 mesh->mVertexBuffers[vCount].set(vertexAlloc);
111 }
112 }
113
114 mesh->mPrimitivesCount = stream->loadU32();
115 if (mesh->mPrimitivesCount) {
116 mesh->mPrimitives = new Primitive_t *[mesh->mPrimitivesCount];
117
118 // load all primitives
119 for (uint32_t pCount = 0; pCount < mesh->mPrimitivesCount; pCount ++) {
120 Primitive_t * prim = new Primitive_t;
121 mesh->mPrimitives[pCount] = prim;
122
123 prim->mPrimitive = (RsPrimitive)stream->loadU8();
124
125 // Check to see if the index buffer was stored
126 uint32_t isIndexPresent = stream->loadU32();
127 if (isIndexPresent) {
128 Allocation *indexAlloc = Allocation::createFromStream(rsc, stream);
129 prim->mIndexBuffer.set(indexAlloc);
130 }
131 }
132 }
133
Alex Sakhartchoukd0f5bd12011-01-31 14:53:24 -0800134#ifndef ANDROID_RS_SERIALIZE
Alex Sakhartchouka3b59602011-01-28 09:31:47 -0800135 mesh->updateGLPrimitives();
136 mesh->initVertexAttribs();
137 mesh->uploadAll(rsc);
138#endif
139 return mesh;
140}
141
Alex Sakhartchoukd0f5bd12011-01-31 14:53:24 -0800142#ifndef ANDROID_RS_SERIALIZE
Alex Sakhartchouka3b59602011-01-28 09:31:47 -0800143
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800144bool Mesh::isValidGLComponent(const Element *elem, uint32_t fieldIdx) {
145 // Do not create attribs for padding
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800146 if (elem->getFieldName(fieldIdx)[0] == '#') {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800147 return false;
148 }
149
150 // Only GL_BYTE, GL_UNSIGNED_BYTE, GL_SHORT, GL_UNSIGNED_SHORT, GL_FIXED, GL_FLOAT are accepted.
151 // Filter rs types accordingly
152 RsDataType dt = elem->getField(fieldIdx)->getComponent().getType();
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800153 if (dt != RS_TYPE_FLOAT_32 && dt != RS_TYPE_UNSIGNED_8 &&
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800154 dt != RS_TYPE_UNSIGNED_16 && dt != RS_TYPE_SIGNED_8 &&
155 dt != RS_TYPE_SIGNED_16) {
156 return false;
157 }
158
159 // Now make sure they are not arrays
160 uint32_t arraySize = elem->getFieldArraySize(fieldIdx);
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800161 if (arraySize != 1) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800162 return false;
163 }
164
165 return true;
166}
167
168void Mesh::initVertexAttribs() {
169 // Count the number of gl attrs to initialize
170 mAttribCount = 0;
171 for (uint32_t ct=0; ct < mVertexBufferCount; ct++) {
172 const Element *elem = mVertexBuffers[ct]->getType()->getElement();
173 for (uint32_t ct=0; ct < elem->getFieldCount(); ct++) {
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800174 if (isValidGLComponent(elem, ct)) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800175 mAttribCount ++;
176 }
177 }
178 }
179
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800180 if (mAttribs) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800181 delete [] mAttribs;
182 delete [] mAttribAllocationIndex;
183 mAttribs = NULL;
184 mAttribAllocationIndex = NULL;
185 }
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800186 if (!mAttribCount) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800187 return;
188 }
189
190 mAttribs = new VertexArray::Attrib[mAttribCount];
191 mAttribAllocationIndex = new uint32_t[mAttribCount];
192
193 uint32_t userNum = 0;
194 for (uint32_t ct=0; ct < mVertexBufferCount; ct++) {
195 const Element *elem = mVertexBuffers[ct]->getType()->getElement();
196 uint32_t stride = elem->getSizeBytes();
197 for (uint32_t fieldI=0; fieldI < elem->getFieldCount(); fieldI++) {
198 const Component &c = elem->getField(fieldI)->getComponent();
199
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800200 if (!isValidGLComponent(elem, fieldI)) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800201 continue;
202 }
203
204 mAttribs[userNum].size = c.getVectorSize();
205 mAttribs[userNum].offset = elem->getFieldOffsetBytes(fieldI);
206 mAttribs[userNum].type = c.getGLType();
207 mAttribs[userNum].normalized = c.getType() != RS_TYPE_FLOAT_32;//c.getIsNormalized();
208 mAttribs[userNum].stride = stride;
209 String8 tmp(RS_SHADER_ATTR);
210 tmp.append(elem->getFieldName(fieldI));
211 mAttribs[userNum].name.setTo(tmp.string());
212
213 // Remember which allocation this attribute came from
214 mAttribAllocationIndex[userNum] = ct;
215 userNum ++;
216 }
217 }
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700218}
219
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800220void Mesh::render(Context *rsc) const {
221 for (uint32_t ct = 0; ct < mPrimitivesCount; ct ++) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700222 renderPrimitive(rsc, ct);
223 }
224}
225
226void Mesh::renderPrimitive(Context *rsc, uint32_t primIndex) const {
227 if (primIndex >= mPrimitivesCount) {
228 LOGE("Invalid primitive index");
229 return;
230 }
231
232 Primitive_t *prim = mPrimitives[primIndex];
233
234 if (prim->mIndexBuffer.get()) {
235 renderPrimitiveRange(rsc, primIndex, 0, prim->mIndexBuffer->getType()->getDimX());
236 return;
237 }
238
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700239 renderPrimitiveRange(rsc, primIndex, 0, mVertexBuffers[0]->getType()->getDimX());
240}
241
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800242void Mesh::renderPrimitiveRange(Context *rsc, uint32_t primIndex, uint32_t start, uint32_t len) const {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800243 if (len < 1 || primIndex >= mPrimitivesCount || mAttribCount == 0) {
244 LOGE("Invalid mesh or parameters");
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700245 return;
246 }
247
248 rsc->checkError("Mesh::renderPrimitiveRange 1");
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700249 for (uint32_t ct=0; ct < mVertexBufferCount; ct++) {
250 mVertexBuffers[ct]->uploadCheck(rsc);
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700251 }
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800252 // update attributes with either buffer information or data ptr based on their current state
253 for (uint32_t ct=0; ct < mAttribCount; ct++) {
254 uint32_t allocIndex = mAttribAllocationIndex[ct];
255 Allocation *alloc = mVertexBuffers[allocIndex].get();
256 if (alloc->getIsBufferObject()) {
257 mAttribs[ct].buffer = alloc->getBufferObjectID();
258 mAttribs[ct].ptr = NULL;
259 } else {
260 mAttribs[ct].buffer = 0;
261 mAttribs[ct].ptr = (const uint8_t*)alloc->getPtr();
262 }
263 }
264
265 VertexArray va(mAttribs, mAttribCount);
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700266 va.setupGL2(rsc, &rsc->mStateVertexArray, &rsc->mShaderCache);
267
268 rsc->checkError("Mesh::renderPrimitiveRange 2");
269 Primitive_t *prim = mPrimitives[primIndex];
270 if (prim->mIndexBuffer.get()) {
271 prim->mIndexBuffer->uploadCheck(rsc);
272 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, prim->mIndexBuffer->getBufferObjectID());
273 glDrawElements(prim->mGLPrimitive, len, GL_UNSIGNED_SHORT, (uint16_t *)(start * 2));
274 } else {
275 glDrawArrays(prim->mGLPrimitive, start, len);
276 }
277
278 rsc->checkError("Mesh::renderPrimitiveRange");
279}
280
281
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800282void Mesh::uploadAll(Context *rsc) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700283 for (uint32_t ct = 0; ct < mVertexBufferCount; ct ++) {
284 if (mVertexBuffers[ct].get()) {
285 mVertexBuffers[ct]->deferedUploadToBufferObject(rsc);
286 }
287 }
288
289 for (uint32_t ct = 0; ct < mPrimitivesCount; ct ++) {
290 if (mPrimitives[ct]->mIndexBuffer.get()) {
291 mPrimitives[ct]->mIndexBuffer->deferedUploadToBufferObject(rsc);
292 }
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700293 }
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700294}
295
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800296void Mesh::updateGLPrimitives() {
297 for (uint32_t i = 0; i < mPrimitivesCount; i ++) {
298 switch (mPrimitives[i]->mPrimitive) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700299 case RS_PRIMITIVE_POINT: mPrimitives[i]->mGLPrimitive = GL_POINTS; break;
300 case RS_PRIMITIVE_LINE: mPrimitives[i]->mGLPrimitive = GL_LINES; break;
301 case RS_PRIMITIVE_LINE_STRIP: mPrimitives[i]->mGLPrimitive = GL_LINE_STRIP; break;
302 case RS_PRIMITIVE_TRIANGLE: mPrimitives[i]->mGLPrimitive = GL_TRIANGLES; break;
303 case RS_PRIMITIVE_TRIANGLE_STRIP: mPrimitives[i]->mGLPrimitive = GL_TRIANGLE_STRIP; break;
304 case RS_PRIMITIVE_TRIANGLE_FAN: mPrimitives[i]->mGLPrimitive = GL_TRIANGLE_FAN; break;
305 }
306 }
Jason Sams7c878f32009-06-30 14:13:04 -0700307}
308
Alex Sakhartchouka80145d2010-08-13 14:32:23 -0700309void Mesh::computeBBox() {
310 float *posPtr = NULL;
311 uint32_t vectorSize = 0;
312 uint32_t stride = 0;
313 uint32_t numVerts = 0;
314 // First we need to find the position ptr and stride
315 for (uint32_t ct=0; ct < mVertexBufferCount; ct++) {
316 const Type *bufferType = mVertexBuffers[ct]->getType();
317 const Element *bufferElem = bufferType->getElement();
318
319 for (uint32_t ct=0; ct < bufferElem->getFieldCount(); ct++) {
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800320 if (strcmp(bufferElem->getFieldName(ct), "position") == 0) {
Alex Sakhartchouka80145d2010-08-13 14:32:23 -0700321 vectorSize = bufferElem->getField(ct)->getComponent().getVectorSize();
322 stride = bufferElem->getSizeBytes() / sizeof(float);
323 uint32_t offset = bufferElem->getFieldOffsetBytes(ct);
324 posPtr = (float*)((uint8_t*)mVertexBuffers[ct]->getPtr() + offset);
325 numVerts = bufferType->getDimX();
326 break;
327 }
328 }
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800329 if (posPtr) {
Alex Sakhartchouka80145d2010-08-13 14:32:23 -0700330 break;
331 }
332 }
333
334 mBBoxMin[0] = mBBoxMin[1] = mBBoxMin[2] = 1e6;
335 mBBoxMax[0] = mBBoxMax[1] = mBBoxMax[2] = -1e6;
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800336 if (!posPtr) {
Alex Sakhartchouka80145d2010-08-13 14:32:23 -0700337 LOGE("Unable to compute bounding box");
338 mBBoxMin[0] = mBBoxMin[1] = mBBoxMin[2] = 0.0f;
339 mBBoxMax[0] = mBBoxMax[1] = mBBoxMax[2] = 0.0f;
340 return;
341 }
342
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800343 for (uint32_t i = 0; i < numVerts; i ++) {
344 for (uint32_t v = 0; v < vectorSize; v ++) {
Alex Sakhartchouka80145d2010-08-13 14:32:23 -0700345 mBBoxMin[v] = rsMin(mBBoxMin[v], posPtr[v]);
346 mBBoxMax[v] = rsMax(mBBoxMax[v], posPtr[v]);
347 }
348 posPtr += stride;
349 }
350}
351
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700352namespace android {
353namespace renderscript {
354
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800355RsMesh rsi_MeshCreate(Context *rsc, uint32_t vtxCount, uint32_t idxCount) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700356 Mesh *sm = new Mesh(rsc);
357 sm->incUserRef();
358
359 sm->mPrimitivesCount = idxCount;
360 sm->mPrimitives = new Mesh::Primitive_t *[sm->mPrimitivesCount];
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800361 for (uint32_t ct = 0; ct < idxCount; ct ++) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700362 sm->mPrimitives[ct] = new Mesh::Primitive_t;
363 }
364
365 sm->mVertexBufferCount = vtxCount;
366 sm->mVertexBuffers = new ObjectBaseRef<Allocation>[vtxCount];
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700367
368 return sm;
369}
370
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800371void rsi_MeshBindVertex(Context *rsc, RsMesh mv, RsAllocation va, uint32_t slot) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700372 Mesh *sm = static_cast<Mesh *>(mv);
373 rsAssert(slot < sm->mVertexBufferCount);
374
375 sm->mVertexBuffers[slot].set((Allocation *)va);
376}
377
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800378void rsi_MeshBindIndex(Context *rsc, RsMesh mv, RsAllocation va, uint32_t primType, uint32_t slot) {
Alex Sakhartchouk164aaed2010-07-01 16:14:06 -0700379 Mesh *sm = static_cast<Mesh *>(mv);
380 rsAssert(slot < sm->mPrimitivesCount);
381
382 sm->mPrimitives[slot]->mIndexBuffer.set((Allocation *)va);
383 sm->mPrimitives[slot]->mPrimitive = (RsPrimitive)primType;
384 sm->updateGLPrimitives();
385}
386
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800387void rsi_MeshInitVertexAttribs(Context *rsc, RsMesh mv) {
Alex Sakhartchouk9d71e212010-11-08 15:10:52 -0800388 Mesh *sm = static_cast<Mesh *>(mv);
389 sm->initVertexAttribs();
390}
391
Alex Sakhartchouk581cc642010-10-27 14:10:07 -0700392}}
393
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800394void rsaMeshGetVertexBufferCount(RsContext con, RsMesh mv, int32_t *numVtx) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700395 Mesh *sm = static_cast<Mesh *>(mv);
396 *numVtx = sm->mVertexBufferCount;
397}
398
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800399void rsaMeshGetIndexCount(RsContext con, RsMesh mv, int32_t *numIdx) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700400 Mesh *sm = static_cast<Mesh *>(mv);
401 *numIdx = sm->mPrimitivesCount;
402}
403
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800404void rsaMeshGetVertices(RsContext con, RsMesh mv, RsAllocation *vtxData, uint32_t vtxDataCount) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700405 Mesh *sm = static_cast<Mesh *>(mv);
406 rsAssert(vtxDataCount == sm->mVertexBufferCount);
407
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800408 for (uint32_t ct = 0; ct < vtxDataCount; ct ++) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700409 vtxData[ct] = sm->mVertexBuffers[ct].get();
410 sm->mVertexBuffers[ct]->incUserRef();
411 }
412}
413
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800414void rsaMeshGetIndices(RsContext con, RsMesh mv, RsAllocation *va, uint32_t *primType, uint32_t idxDataCount) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700415 Mesh *sm = static_cast<Mesh *>(mv);
416 rsAssert(idxDataCount == sm->mPrimitivesCount);
417
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800418 for (uint32_t ct = 0; ct < idxDataCount; ct ++) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700419 va[ct] = sm->mPrimitives[ct]->mIndexBuffer.get();
420 primType[ct] = sm->mPrimitives[ct]->mPrimitive;
Alex Sakhartchouked9f2102010-11-09 17:00:54 -0800421 if (sm->mPrimitives[ct]->mIndexBuffer.get()) {
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700422 sm->mPrimitives[ct]->mIndexBuffer->incUserRef();
423 }
424 }
Alex Sakhartchouk80a4c2c2010-07-12 15:50:32 -0700425}
Alex Sakhartchouka3b59602011-01-28 09:31:47 -0800426
427#endif