/* * Copyright (C) 2009 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "rsContext.h" #include #include "rsFileA3D.h" #include "rsMesh.h" using namespace android; using namespace android::renderscript; FileA3D::FileA3D() { mRsc = NULL; } FileA3D::~FileA3D() { } bool FileA3D::load(Context *rsc, FILE *f) { char magicString[12]; size_t len; LOGE("file open 1"); len = fread(magicString, 1, 12, f); if ((len != 12) || memcmp(magicString, "Android3D_ff", 12)) { return false; } LOGE("file open 2"); len = fread(&mMajorVersion, 1, sizeof(mMajorVersion), f); if (len != sizeof(mMajorVersion)) { return false; } LOGE("file open 3"); len = fread(&mMinorVersion, 1, sizeof(mMinorVersion), f); if (len != sizeof(mMinorVersion)) { return false; } LOGE("file open 4"); uint32_t flags; len = fread(&flags, 1, sizeof(flags), f); if (len != sizeof(flags)) { return false; } mUse64BitOffsets = (flags & 1) != 0; LOGE("file open 64bit = %i", mUse64BitOffsets); if (mUse64BitOffsets) { len = fread(&mDataSize, 1, sizeof(mDataSize), f); if (len != sizeof(mDataSize)) { return false; } } else { uint32_t tmp; len = fread(&tmp, 1, sizeof(tmp), f); if (len != sizeof(tmp)) { return false; } mDataSize = tmp; } LOGE("file open size = %lli", mDataSize); // We should know enough to read the file in at this point. fseek(f, SEEK_SET, 0); mAlloc= malloc(mDataSize); if (!mAlloc) { return false; } mData = (uint8_t *)mAlloc; len = fread(mAlloc, 1, mDataSize, f); if (len != mDataSize) { return false; } LOGE("file start processing"); return process(rsc); } bool FileA3D::processIndex(Context *rsc, A3DIndexEntry *ie) { bool ret = false; IO io(mData + ie->mOffset, mUse64BitOffsets); LOGE("process index, type %i", ie->mType); switch(ie->mType) { case CHUNK_ELEMENT: processChunk_Element(rsc, &io, ie); break; case CHUNK_ELEMENT_SOURCE: processChunk_ElementSource(rsc, &io, ie); break; case CHUNK_VERTICIES: processChunk_Verticies(rsc, &io, ie); break; case CHUNK_MESH: processChunk_Mesh(rsc, &io, ie); break; case CHUNK_PRIMITIVE: processChunk_Primitive(rsc, &io, ie); break; default: LOGE("FileA3D Unknown chunk type"); break; } return (ie->mRsObj != NULL); } bool FileA3D::process(Context *rsc) { LOGE("process"); IO io(mData + 12, mUse64BitOffsets); bool ret = true; // Build the index first LOGE("process 1"); io.loadU32(); // major version, already loaded io.loadU32(); // minor version, already loaded LOGE("process 2"); io.loadU32(); // flags io.loadOffset(); // filesize, already loaded. LOGE("process 4"); uint64_t mIndexOffset = io.loadOffset(); uint64_t mStringOffset = io.loadOffset(); LOGE("process mIndexOffset= 0x%016llx", mIndexOffset); LOGE("process mStringOffset= 0x%016llx", mStringOffset); IO index(mData + mIndexOffset, mUse64BitOffsets); IO stringTable(mData + mStringOffset, mUse64BitOffsets); uint32_t stringEntryCount = stringTable.loadU32(); LOGE("stringEntryCount %i", stringEntryCount); mStrings.setCapacity(stringEntryCount); mStringIndexValues.setCapacity(stringEntryCount); if (stringEntryCount) { uint32_t stringType = stringTable.loadU32(); LOGE("stringType %i", stringType); rsAssert(stringType==0); for (uint32_t ct = 0; ct < stringEntryCount; ct++) { uint64_t offset = stringTable.loadOffset(); LOGE("string offset 0x%016llx", offset); IO tmp(mData + offset, mUse64BitOffsets); String8 s; tmp.loadString(&s); LOGE("string %s", s.string()); mStrings.push(s); } } LOGE("strings done"); uint32_t indexEntryCount = index.loadU32(); LOGE("index count %i", indexEntryCount); mIndex.setCapacity(indexEntryCount); for (uint32_t ct = 0; ct < indexEntryCount; ct++) { A3DIndexEntry e; uint32_t stringIndex = index.loadU32(); LOGE("index %i", ct); LOGE(" string index %i", stringIndex); e.mType = (A3DChunkType)index.loadU32(); LOGE(" type %i", e.mType); e.mOffset = index.loadOffset(); LOGE(" offset 0x%016llx", e.mOffset); if (stringIndex && (stringIndex < mStrings.size())) { e.mID = mStrings[stringIndex]; mStringIndexValues.editItemAt(stringIndex) = ct; LOGE(" id %s", e.mID.string()); } mIndex.push(e); } LOGE("index done"); // At this point the index should be fully populated. // We can now walk though it and load all the objects. for (uint32_t ct = 0; ct < indexEntryCount; ct++) { LOGE("processing index entry %i", ct); processIndex(rsc, &mIndex.editItemAt(ct)); } return ret; } FileA3D::IO::IO(const uint8_t *buf, bool use64) { mData = buf; mPos = 0; mUse64 = use64; } uint64_t FileA3D::IO::loadOffset() { uint64_t tmp; if (mUse64) { mPos = (mPos + 7) & (~7); tmp = reinterpret_cast(&mData[mPos])[0]; mPos += sizeof(uint64_t); return tmp; } return loadU32(); } void FileA3D::IO::loadString(String8 *s) { LOGE("loadString"); uint32_t len = loadU32(); LOGE("loadString len %i", len); s->setTo((const char *)&mData[mPos], len); mPos += len; } void FileA3D::processChunk_Mesh(Context *rsc, IO *io, A3DIndexEntry *ie) { Mesh * m = new Mesh(rsc); m->mPrimitivesCount = io->loadU32(); m->mPrimitives = new Mesh::Primitive_t *[m->mPrimitivesCount]; for (uint32_t ct = 0; ct < m->mPrimitivesCount; ct++) { uint32_t index = io->loadU32(); m->mPrimitives[ct] = (Mesh::Primitive_t *)mIndex[index].mRsObj; } ie->mRsObj = m; } void FileA3D::processChunk_Primitive(Context *rsc, IO *io, A3DIndexEntry *ie) { Mesh::Primitive_t * p = new Mesh::Primitive_t; p->mIndexCount = io->loadU32(); uint32_t vertIdx = io->loadU32(); p->mRestartCounts = io->loadU16(); uint32_t bits = io->loadU8(); p->mType = (RsPrimitive)io->loadU8(); LOGE("processChunk_Primitive count %i, bits %i", p->mIndexCount, bits); p->mVerticies = (Mesh::Verticies_t *)mIndex[vertIdx].mRsObj; p->mIndicies = new uint16_t[p->mIndexCount]; for (uint32_t ct = 0; ct < p->mIndexCount; ct++) { switch(bits) { case 8: p->mIndicies[ct] = io->loadU8(); break; case 16: p->mIndicies[ct] = io->loadU16(); break; case 32: p->mIndicies[ct] = io->loadU32(); break; } LOGE(" idx %i", p->mIndicies[ct]); } if (p->mRestartCounts) { p->mRestarts = new uint16_t[p->mRestartCounts]; for (uint32_t ct = 0; ct < p->mRestartCounts; ct++) { switch(bits) { case 8: p->mRestarts[ct] = io->loadU8(); break; case 16: p->mRestarts[ct] = io->loadU16(); break; case 32: p->mRestarts[ct] = io->loadU32(); break; } LOGE(" idx %i", p->mRestarts[ct]); } } else { p->mRestarts = NULL; } ie->mRsObj = p; } void FileA3D::processChunk_Verticies(Context *rsc, IO *io, A3DIndexEntry *ie) { Mesh::Verticies_t *cv = new Mesh::Verticies_t; cv->mAllocationCount = io->loadU32(); cv->mAllocations = new Allocation *[cv->mAllocationCount]; LOGE("processChunk_Verticies count %i", cv->mAllocationCount); for (uint32_t ct = 0; ct < cv->mAllocationCount; ct++) { uint32_t i = io->loadU32(); cv->mAllocations[ct] = (Allocation *)mIndex[i].mRsObj; LOGE(" idx %i", i); } ie->mRsObj = cv; } void FileA3D::processChunk_Element(Context *rsc, IO *io, A3DIndexEntry *ie) { /* rsi_ElementBegin(rsc); uint32_t count = io->loadU32(); LOGE("processChunk_Element count %i", count); while (count--) { RsDataKind dk = (RsDataKind)io->loadU8(); RsDataType dt = (RsDataType)io->loadU8(); uint32_t bits = io->loadU8(); bool isNorm = io->loadU8() != 0; LOGE(" %i %i %i %i", dk, dt, bits, isNorm); rsi_ElementAdd(rsc, dk, dt, isNorm, bits, 0); } LOGE("processChunk_Element create"); ie->mRsObj = rsi_ElementCreate(rsc); */ } void FileA3D::processChunk_ElementSource(Context *rsc, IO *io, A3DIndexEntry *ie) { uint32_t index = io->loadU32(); uint32_t count = io->loadU32(); LOGE("processChunk_ElementSource count %i, index %i", count, index); RsElement e = (RsElement)mIndex[index].mRsObj; RsAllocation a = rsi_AllocationCreateSized(rsc, e, count); Allocation * alloc = static_cast(a); float * data = (float *)alloc->getPtr(); while(count--) { *data = io->loadF(); LOGE(" %f", *data); data++; } ie->mRsObj = alloc; } namespace android { namespace renderscript { RsFile rsi_FileOpen(Context *rsc, char const *path, unsigned int len) { FileA3D *fa3d = new FileA3D; FILE *f = fopen("/sdcard/test.a3d", "rb"); if (f) { fa3d->load(rsc, f); fclose(f); return fa3d; } delete fa3d; return NULL; } } }