blob: 3b963fe542c601296a2bd5de6d823b1c65bb8294 [file] [log] [blame]
Jason Samsa5597fc2009-07-08 18:01:53 -07001
2/*
3 * Copyright (C) 2009 The Android Open Source Project
4 *
5 * Licensed under the Apache License, Version 2.0 (the "License");
6 * you may not use this file except in compliance with the License.
7 * You may obtain a copy of the License at
8 *
9 * http://www.apache.org/licenses/LICENSE-2.0
10 *
11 * Unless required by applicable law or agreed to in writing, software
12 * distributed under the License is distributed on an "AS IS" BASIS,
13 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14 * See the License for the specific language governing permissions and
15 * limitations under the License.
16 */
17
18#include "rsContext.h"
Jason Samsa5597fc2009-07-08 18:01:53 -070019#include "rsFileA3D.h"
20
21#include "rsMesh.h"
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070022#include "rsAnimation.h"
Alex Sakhartchouke23d2392012-03-09 09:24:39 -080023#include "rs.h"
Alex Sakhartchoukd3e0ad42010-06-24 17:15:34 -070024
Jason Samsa6ab26a2012-03-28 17:44:21 -070025#if !defined(__RS_PDK__)
26 #include <androidfw/Asset.h>
27#endif
28
Jason Samsa5597fc2009-07-08 18:01:53 -070029using namespace android;
30using namespace android::renderscript;
31
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080032FileA3D::FileA3D(Context *rsc) : ObjectBase(rsc) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070033 mAlloc = NULL;
34 mData = NULL;
35 mWriteStream = NULL;
36 mReadStream = NULL;
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080037 mAsset = NULL;
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070038
39 mMajorVersion = 0;
40 mMinorVersion = 1;
41 mDataSize = 0;
Jason Samsa5597fc2009-07-08 18:01:53 -070042}
43
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080044FileA3D::~FileA3D() {
45 for (size_t i = 0; i < mIndex.size(); i ++) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070046 delete mIndex[i];
47 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080048 for (size_t i = 0; i < mWriteIndex.size(); i ++) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070049 delete mWriteIndex[i];
50 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080051 if (mWriteStream) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070052 delete mWriteStream;
53 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080054 if (mReadStream) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070055 delete mWriteStream;
56 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080057 if (mAlloc) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -070058 free(mAlloc);
59 }
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080060 if (mAsset) {
Jason Samsa6ab26a2012-03-28 17:44:21 -070061#if !defined(__RS_PDK__)
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080062 delete mAsset;
Jason Samsa6ab26a2012-03-28 17:44:21 -070063#endif
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080064 }
Jason Samsa5597fc2009-07-08 18:01:53 -070065}
66
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080067void FileA3D::parseHeader(IStream *headerStream) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070068 mMajorVersion = headerStream->loadU32();
69 mMinorVersion = headerStream->loadU32();
70 uint32_t flags = headerStream->loadU32();
71 mUse64BitOffsets = (flags & 1) != 0;
72
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070073 uint32_t numIndexEntries = headerStream->loadU32();
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080074 for (uint32_t i = 0; i < numIndexEntries; i ++) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070075 A3DIndexEntry *entry = new A3DIndexEntry();
Jason Sams48ecf6a2013-07-09 15:35:29 -070076 entry->mObjectName = headerStream->loadString();
77
Steve Block65982012011-10-20 11:56:00 +010078 //ALOGV("Header data, entry name = %s", entry->mObjectName.string());
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070079 entry->mType = (RsA3DClassID)headerStream->loadU32();
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080080 if (mUse64BitOffsets){
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070081 entry->mOffset = headerStream->loadOffset();
82 entry->mLength = headerStream->loadOffset();
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -080083 } else {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -070084 entry->mOffset = headerStream->loadU32();
85 entry->mLength = headerStream->loadU32();
86 }
87 entry->mRsObj = NULL;
88 mIndex.push(entry);
89 }
90}
91
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080092bool FileA3D::load(Asset *asset) {
Jason Samsa6ab26a2012-03-28 17:44:21 -070093#if !defined(__RS_PDK__)
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080094 mAsset = asset;
95 return load(asset->getBuffer(false), asset->getLength());
Jason Samsa6ab26a2012-03-28 17:44:21 -070096#else
97 return false;
98#endif
Alex Sakhartchouk5224a272011-01-07 11:12:08 -080099}
100
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800101bool FileA3D::load(const void *data, size_t length) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700102 const uint8_t *localData = (const uint8_t *)data;
103
104 size_t lengthRemaining = length;
105 size_t magicStrLen = 12;
106 if ((length < magicStrLen) ||
107 memcmp(data, "Android3D_ff", magicStrLen)) {
108 return false;
109 }
110
111 localData += magicStrLen;
112 lengthRemaining -= magicStrLen;
113
114 // Next we get our header size
115 uint64_t headerSize = 0;
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800116 if (lengthRemaining < sizeof(headerSize)) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700117 return false;
118 }
119
120 memcpy(&headerSize, localData, sizeof(headerSize));
121 localData += sizeof(headerSize);
122 lengthRemaining -= sizeof(headerSize);
123
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800124 if (lengthRemaining < headerSize) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700125 return false;
126 }
127
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700128 // Now open the stream to parse the header
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700129 IStream headerStream(localData, false);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700130 parseHeader(&headerStream);
131
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700132 localData += headerSize;
133 lengthRemaining -= headerSize;
134
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800135 if (lengthRemaining < sizeof(mDataSize)) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700136 return false;
137 }
138
139 // Read the size of the data
140 memcpy(&mDataSize, localData, sizeof(mDataSize));
141 localData += sizeof(mDataSize);
142 lengthRemaining -= sizeof(mDataSize);
143
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800144 if (lengthRemaining < mDataSize) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700145 return false;
146 }
147
148 // We should know enough to read the file in at this point.
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700149 mData = (uint8_t *)localData;
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700150 mReadStream = new IStream(mData, mUse64BitOffsets);
151
152 return true;
153}
154
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800155bool FileA3D::load(FILE *f) {
Jason Samsa5597fc2009-07-08 18:01:53 -0700156 char magicString[12];
157 size_t len;
158
Steve Block65982012011-10-20 11:56:00 +0100159 ALOGV("file open 1");
Jason Samsa5597fc2009-07-08 18:01:53 -0700160 len = fread(magicString, 1, 12, f);
161 if ((len != 12) ||
162 memcmp(magicString, "Android3D_ff", 12)) {
163 return false;
164 }
165
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700166 // Next thing is the size of the header
167 uint64_t headerSize = 0;
168 len = fread(&headerSize, 1, sizeof(headerSize), f);
169 if (len != sizeof(headerSize) || headerSize == 0) {
Jason Samsa5597fc2009-07-08 18:01:53 -0700170 return false;
171 }
172
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700173 uint8_t *headerData = (uint8_t *)malloc(headerSize);
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800174 if (!headerData) {
Jason Samsa5597fc2009-07-08 18:01:53 -0700175 return false;
176 }
177
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700178 len = fread(headerData, 1, headerSize, f);
179 if (len != headerSize) {
Jason Samsa5597fc2009-07-08 18:01:53 -0700180 return false;
181 }
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700182
183 // Now open the stream to parse the header
184 IStream headerStream(headerData, false);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700185 parseHeader(&headerStream);
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700186
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700187 free(headerData);
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700188
189 // Next thing is the size of the header
190 len = fread(&mDataSize, 1, sizeof(mDataSize), f);
191 if (len != sizeof(mDataSize) || mDataSize == 0) {
192 return false;
Jason Samsa5597fc2009-07-08 18:01:53 -0700193 }
194
Steve Block65982012011-10-20 11:56:00 +0100195 ALOGV("file open size = %lli", mDataSize);
Jason Samsa5597fc2009-07-08 18:01:53 -0700196
197 // We should know enough to read the file in at this point.
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700198 mAlloc = malloc(mDataSize);
Jason Samsa5597fc2009-07-08 18:01:53 -0700199 if (!mAlloc) {
200 return false;
201 }
202 mData = (uint8_t *)mAlloc;
203 len = fread(mAlloc, 1, mDataSize, f);
204 if (len != mDataSize) {
205 return false;
206 }
207
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700208 mReadStream = new IStream(mData, mUse64BitOffsets);
209
Steve Block65982012011-10-20 11:56:00 +0100210 ALOGV("Header is read an stream initialized");
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700211 return true;
Jason Samsa5597fc2009-07-08 18:01:53 -0700212}
213
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700214size_t FileA3D::getNumIndexEntries() const {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700215 return mIndex.size();
216}
Jason Samsa5597fc2009-07-08 18:01:53 -0700217
Jason Sams48ecf6a2013-07-09 15:35:29 -0700218FileA3D::A3DIndexEntry::~A3DIndexEntry() {
219 delete[] mObjectName;
220}
221
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700222const FileA3D::A3DIndexEntry *FileA3D::getIndexEntry(size_t index) const {
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800223 if (index < mIndex.size()) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700224 return mIndex[index];
Jason Samsa5597fc2009-07-08 18:01:53 -0700225 }
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700226 return NULL;
Jason Samsa5597fc2009-07-08 18:01:53 -0700227}
228
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700229ObjectBase *FileA3D::initializeFromEntry(size_t index) {
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800230 if (index >= mIndex.size()) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700231 return NULL;
232 }
233
234 FileA3D::A3DIndexEntry *entry = mIndex[index];
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800235 if (!entry) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700236 return NULL;
237 }
238
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800239 if (entry->mRsObj) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700240 entry->mRsObj->incUserRef();
241 return entry->mRsObj;
242 }
243
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700244 // Seek to the beginning of object
245 mReadStream->reset(entry->mOffset);
246 switch (entry->mType) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700247 case RS_A3D_CLASS_ID_UNKNOWN:
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700248 return NULL;
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700249 case RS_A3D_CLASS_ID_MESH:
250 entry->mRsObj = Mesh::createFromStream(mRSC, mReadStream);
251 break;
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700252 case RS_A3D_CLASS_ID_TYPE:
253 entry->mRsObj = Type::createFromStream(mRSC, mReadStream);
254 break;
255 case RS_A3D_CLASS_ID_ELEMENT:
256 entry->mRsObj = Element::createFromStream(mRSC, mReadStream);
257 break;
258 case RS_A3D_CLASS_ID_ALLOCATION:
259 entry->mRsObj = Allocation::createFromStream(mRSC, mReadStream);
260 break;
261 case RS_A3D_CLASS_ID_PROGRAM_VERTEX:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800262 //entry->mRsObj = ProgramVertex::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700263 break;
264 case RS_A3D_CLASS_ID_PROGRAM_RASTER:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800265 //entry->mRsObj = ProgramRaster::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700266 break;
267 case RS_A3D_CLASS_ID_PROGRAM_FRAGMENT:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800268 //entry->mRsObj = ProgramFragment::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700269 break;
270 case RS_A3D_CLASS_ID_PROGRAM_STORE:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800271 //entry->mRsObj = ProgramStore::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700272 break;
273 case RS_A3D_CLASS_ID_SAMPLER:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800274 //entry->mRsObj = Sampler::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700275 break;
276 case RS_A3D_CLASS_ID_ANIMATION:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800277 //entry->mRsObj = Animation::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700278 break;
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700279 case RS_A3D_CLASS_ID_ADAPTER_1D:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800280 //entry->mRsObj = Adapter1D::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700281 break;
282 case RS_A3D_CLASS_ID_ADAPTER_2D:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800283 //entry->mRsObj = Adapter2D::createFromStream(mRSC, mReadStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700284 break;
285 case RS_A3D_CLASS_ID_SCRIPT_C:
Alex Sakhartchouk099d7d32011-01-28 09:31:47 -0800286 break;
Jason Samsdbe66d62012-09-17 13:54:41 -0700287 case RS_A3D_CLASS_ID_SCRIPT_KERNEL_ID:
288 break;
289 case RS_A3D_CLASS_ID_SCRIPT_FIELD_ID:
290 break;
291 case RS_A3D_CLASS_ID_SCRIPT_METHOD_ID:
292 break;
293 case RS_A3D_CLASS_ID_SCRIPT_GROUP:
294 break;
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700295 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800296 if (entry->mRsObj) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700297 entry->mRsObj->incUserRef();
298 }
299 return entry->mRsObj;
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700300}
301
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800302bool FileA3D::writeFile(const char *filename) {
303 if (!mWriteStream) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000304 ALOGE("No objects to write\n");
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700305 return false;
306 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800307 if (mWriteStream->getPos() == 0) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000308 ALOGE("No objects to write\n");
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700309 return false;
310 }
Jason Samsa5597fc2009-07-08 18:01:53 -0700311
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700312 FILE *writeHandle = fopen(filename, "wb");
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800313 if (!writeHandle) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000314 ALOGE("Couldn't open the file for writing\n");
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700315 return false;
316 }
Jason Samsa5597fc2009-07-08 18:01:53 -0700317
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700318 // Open a new stream to make writing the header easier
319 OStream headerStream(5*1024, false);
320 headerStream.addU32(mMajorVersion);
321 headerStream.addU32(mMinorVersion);
322 uint32_t is64Bit = 0;
323 headerStream.addU32(is64Bit);
Jason Samsa5597fc2009-07-08 18:01:53 -0700324
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700325 uint32_t writeIndexSize = mWriteIndex.size();
326 headerStream.addU32(writeIndexSize);
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800327 for (uint32_t i = 0; i < writeIndexSize; i ++) {
Jason Sams48ecf6a2013-07-09 15:35:29 -0700328 headerStream.addString(mWriteIndex[i]->mObjectName);
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700329 headerStream.addU32((uint32_t)mWriteIndex[i]->mType);
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800330 if (mUse64BitOffsets){
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700331 headerStream.addOffset(mWriteIndex[i]->mOffset);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700332 headerStream.addOffset(mWriteIndex[i]->mLength);
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800333 } else {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700334 uint32_t offset = (uint32_t)mWriteIndex[i]->mOffset;
335 headerStream.addU32(offset);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700336 offset = (uint32_t)mWriteIndex[i]->mLength;
337 headerStream.addU32(offset);
Jason Samsa5597fc2009-07-08 18:01:53 -0700338 }
339 }
340
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700341 // Write our magic string so we know we are reading the right file
Jason Sams48ecf6a2013-07-09 15:35:29 -0700342 fwrite(A3D_MAGIC_KEY, sizeof(char), strlen(A3D_MAGIC_KEY), writeHandle);
Jason Samsa5597fc2009-07-08 18:01:53 -0700343
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700344 // Store the size of the header to make it easier to parse when we read it
345 uint64_t headerSize = headerStream.getPos();
346 fwrite(&headerSize, sizeof(headerSize), 1, writeHandle);
Jason Samsa5597fc2009-07-08 18:01:53 -0700347
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700348 // Now write our header
349 fwrite(headerStream.getPtr(), sizeof(uint8_t), headerStream.getPos(), writeHandle);
Jason Samsa5597fc2009-07-08 18:01:53 -0700350
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700351 // Now write the size of the data part of the file for easier parsing later
352 uint64_t fileDataSize = mWriteStream->getPos();
353 fwrite(&fileDataSize, sizeof(fileDataSize), 1, writeHandle);
354
355 fwrite(mWriteStream->getPtr(), sizeof(uint8_t), mWriteStream->getPos(), writeHandle);
356
357 int status = fclose(writeHandle);
358
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800359 if (status != 0) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000360 ALOGE("Couldn't close file\n");
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700361 return false;
Jason Samsa5597fc2009-07-08 18:01:53 -0700362 }
363
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700364 return true;
Jason Samsa5597fc2009-07-08 18:01:53 -0700365}
366
Jason Samse3150cf2012-07-24 18:10:20 -0700367void FileA3D::appendToFile(Context *con, ObjectBase *obj) {
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800368 if (!obj) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700369 return;
Jason Samsa5597fc2009-07-08 18:01:53 -0700370 }
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800371 if (!mWriteStream) {
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700372 const uint64_t initialStreamSize = 256*1024;
373 mWriteStream = new OStream(initialStreamSize, false);
Jason Samsa5597fc2009-07-08 18:01:53 -0700374 }
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700375 A3DIndexEntry *indexEntry = new A3DIndexEntry();
Jason Sams48ecf6a2013-07-09 15:35:29 -0700376 indexEntry->mObjectName = rsuCopyString(obj->getName());
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700377 indexEntry->mType = obj->getClassId();
378 indexEntry->mOffset = mWriteStream->getPos();
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700379 indexEntry->mRsObj = obj;
Alex Sakhartchoukfb6b6142010-05-21 12:53:13 -0700380 mWriteIndex.push(indexEntry);
Jason Samse3150cf2012-07-24 18:10:20 -0700381 obj->serialize(con, mWriteStream);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700382 indexEntry->mLength = mWriteStream->getPos() - indexEntry->mOffset;
383 mWriteStream->align(4);
Jason Samsa5597fc2009-07-08 18:01:53 -0700384}
385
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800386RsObjectBase rsaFileA3DGetEntryByIndex(RsContext con, uint32_t index, RsFile file) {
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700387 FileA3D *fa3d = static_cast<FileA3D *>(file);
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800388 if (!fa3d) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000389 ALOGE("Can't load entry. No valid file");
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700390 return NULL;
391 }
392
393 ObjectBase *obj = fa3d->initializeFromEntry(index);
Steve Block65982012011-10-20 11:56:00 +0100394 //ALOGV("Returning object with name %s", obj->getName());
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700395
396 return obj;
397}
398
399
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800400void rsaFileA3DGetNumIndexEntries(RsContext con, int32_t *numEntries, RsFile file) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700401 FileA3D *fa3d = static_cast<FileA3D *>(file);
402
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800403 if (fa3d) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700404 *numEntries = fa3d->getNumIndexEntries();
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800405 } else {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700406 *numEntries = 0;
407 }
408}
409
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800410void rsaFileA3DGetIndexEntries(RsContext con, RsFileIndexEntry *fileEntries, uint32_t numEntries, RsFile file) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700411 FileA3D *fa3d = static_cast<FileA3D *>(file);
412
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800413 if (!fa3d) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000414 ALOGE("Can't load index entries. No valid file");
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700415 return;
416 }
417
418 uint32_t numFileEntries = fa3d->getNumIndexEntries();
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800419 if (numFileEntries != numEntries || numEntries == 0 || fileEntries == NULL) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000420 ALOGE("Can't load index entries. Invalid number requested");
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700421 return;
422 }
423
Alex Sakhartchoukafb743a2010-11-09 17:00:54 -0800424 for (uint32_t i = 0; i < numFileEntries; i ++) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700425 const FileA3D::A3DIndexEntry *entry = fa3d->getIndexEntry(i);
426 fileEntries[i].classID = entry->getType();
Jason Sams48ecf6a2013-07-09 15:35:29 -0700427 fileEntries[i].objectName = rsuCopyString(entry->getObjectName());
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700428 }
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700429}
430
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800431RsFile rsaFileA3DCreateFromMemory(RsContext con, const void *data, uint32_t len) {
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700432 if (data == NULL) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000433 ALOGE("File load failed. Asset stream is NULL");
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700434 return NULL;
435 }
436
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700437 Context *rsc = static_cast<Context *>(con);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700438 FileA3D *fa3d = new FileA3D(rsc);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700439 fa3d->incUserRef();
440
Alex Sakhartchoukdc763f32010-10-27 14:10:07 -0700441 fa3d->load(data, len);
Alex Sakhartchoukb825f672010-06-04 10:06:50 -0700442 return fa3d;
443}
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800444
445RsFile rsaFileA3DCreateFromAsset(RsContext con, void *_asset) {
Jason Samsa6ab26a2012-03-28 17:44:21 -0700446#if !defined(__RS_PDK__)
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800447 Context *rsc = static_cast<Context *>(con);
448 Asset *asset = static_cast<Asset *>(_asset);
449 FileA3D *fa3d = new FileA3D(rsc);
450 fa3d->incUserRef();
451
452 fa3d->load(asset);
453 return fa3d;
Jason Samsa6ab26a2012-03-28 17:44:21 -0700454#else
455 return NULL;
456#endif
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800457}
458
459RsFile rsaFileA3DCreateFromFile(RsContext con, const char *path) {
460 if (path == NULL) {
Steve Blockaf12ac62012-01-06 19:20:56 +0000461 ALOGE("File load failed. Path is NULL");
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800462 return NULL;
463 }
464
465 Context *rsc = static_cast<Context *>(con);
466 FileA3D *fa3d = NULL;
467
468 FILE *f = fopen(path, "rb");
469 if (f) {
470 fa3d = new FileA3D(rsc);
471 fa3d->incUserRef();
472 fa3d->load(f);
473 fclose(f);
474 } else {
Steve Blockaf12ac62012-01-06 19:20:56 +0000475 ALOGE("Could not open file %s", path);
Alex Sakhartchouk5224a272011-01-07 11:12:08 -0800476 }
477
478 return fa3d;
479}