1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501 |
- /*
- ---------------------------------------------------------------------------
- Open Asset Import Library (assimp)
- ---------------------------------------------------------------------------
- Copyright (c) 2006-2018, assimp team
- All rights reserved.
- Redistribution and use of this software in source and binary forms,
- with or without modification, are permitted provided that the following
- conditions are met:
- * Redistributions of source code must retain the above
- copyright notice, this list of conditions and the
- following disclaimer.
- * Redistributions in binary form must reproduce the above
- copyright notice, this list of conditions and the
- following disclaimer in the documentation and/or other
- materials provided with the distribution.
- * Neither the name of the assimp team, nor the names of its
- contributors may be used to endorse or promote products
- derived from this software without specific prior
- written permission of the assimp team.
- THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- ---------------------------------------------------------------------------
- */
- /** @file Implementation of the XFile parser helper class */
- #ifndef ASSIMP_BUILD_NO_X_IMPORTER
- #include "XFileParser.h"
- #include "XFileHelper.h"
- #include <assimp/fast_atof.h>
- #include <assimp/Exceptional.h>
- #include <assimp/TinyFormatter.h>
- #include <assimp/ByteSwapper.h>
- #include <assimp/StringUtils.h>
- #include <assimp/DefaultLogger.hpp>
- using namespace Assimp;
- using namespace Assimp::XFile;
- using namespace Assimp::Formatter;
- #ifndef ASSIMP_BUILD_NO_COMPRESSED_X
- # ifdef ASSIMP_BUILD_NO_OWN_ZLIB
- # include <zlib.h>
- # else
- # include "../contrib/zlib/zlib.h"
- # endif
- // Magic identifier for MSZIP compressed data
- #define MSZIP_MAGIC 0x4B43
- #define MSZIP_BLOCK 32786
- // ------------------------------------------------------------------------------------------------
- // Dummy memory wrappers for use with zlib
- static void* dummy_alloc (void* /*opaque*/, unsigned int items, unsigned int size) {
- return ::operator new(items*size);
- }
- static void dummy_free (void* /*opaque*/, void* address) {
- return ::operator delete(address);
- }
- #endif // !! ASSIMP_BUILD_NO_COMPRESSED_X
- // ------------------------------------------------------------------------------------------------
- // Constructor. Creates a data structure out of the XFile given in the memory block.
- XFileParser::XFileParser( const std::vector<char>& pBuffer)
- : mMajorVersion( 0 )
- , mMinorVersion( 0 )
- , mIsBinaryFormat( false )
- , mBinaryNumCount( 0 )
- , mP( nullptr )
- , mEnd( nullptr )
- , mLineNumber( 0 )
- , mScene( nullptr ) {
- // vector to store uncompressed file for INFLATE'd X files
- std::vector<char> uncompressed;
- // set up memory pointers
- mP = &pBuffer.front();
- mEnd = mP + pBuffer.size() - 1;
- // check header
- if ( 0 != strncmp( mP, "xof ", 4 ) ) {
- throw DeadlyImportError( "Header mismatch, file is not an XFile." );
- }
- // read version. It comes in a four byte format such as "0302"
- mMajorVersion = (unsigned int)(mP[4] - 48) * 10 + (unsigned int)(mP[5] - 48);
- mMinorVersion = (unsigned int)(mP[6] - 48) * 10 + (unsigned int)(mP[7] - 48);
- bool compressed = false;
- // txt - pure ASCII text format
- if( strncmp( mP + 8, "txt ", 4) == 0)
- mIsBinaryFormat = false;
- // bin - Binary format
- else if( strncmp( mP + 8, "bin ", 4) == 0)
- mIsBinaryFormat = true;
- // tzip - Inflate compressed text format
- else if( strncmp( mP + 8, "tzip", 4) == 0)
- {
- mIsBinaryFormat = false;
- compressed = true;
- }
- // bzip - Inflate compressed binary format
- else if( strncmp( mP + 8, "bzip", 4) == 0)
- {
- mIsBinaryFormat = true;
- compressed = true;
- }
- else ThrowException( format() << "Unsupported xfile format '" <<
- mP[8] << mP[9] << mP[10] << mP[11] << "'");
- // float size
- mBinaryFloatSize = (unsigned int)(mP[12] - 48) * 1000
- + (unsigned int)(mP[13] - 48) * 100
- + (unsigned int)(mP[14] - 48) * 10
- + (unsigned int)(mP[15] - 48);
- if( mBinaryFloatSize != 32 && mBinaryFloatSize != 64)
- ThrowException( format() << "Unknown float size " << mBinaryFloatSize << " specified in xfile header." );
- // The x format specifies size in bits, but we work in bytes
- mBinaryFloatSize /= 8;
- mP += 16;
- // If this is a compressed X file, apply the inflate algorithm to it
- if (compressed)
- {
- #ifdef ASSIMP_BUILD_NO_COMPRESSED_X
- throw DeadlyImportError("Assimp was built without compressed X support");
- #else
- /* ///////////////////////////////////////////////////////////////////////
- * COMPRESSED X FILE FORMAT
- * ///////////////////////////////////////////////////////////////////////
- * [xhead]
- * 2 major
- * 2 minor
- * 4 type // bzip,tzip
- * [mszip_master_head]
- * 4 unkn // checksum?
- * 2 unkn // flags? (seems to be constant)
- * [mszip_head]
- * 2 ofs // offset to next section
- * 2 magic // 'CK'
- * ... ofs bytes of data
- * ... next mszip_head
- *
- * http://www.kdedevelopers.org/node/3181 has been very helpful.
- * ///////////////////////////////////////////////////////////////////////
- */
- // build a zlib stream
- z_stream stream;
- stream.opaque = NULL;
- stream.zalloc = &dummy_alloc;
- stream.zfree = &dummy_free;
- stream.data_type = (mIsBinaryFormat ? Z_BINARY : Z_ASCII);
- // initialize the inflation algorithm
- ::inflateInit2(&stream, -MAX_WBITS);
- // skip unknown data (checksum, flags?)
- mP += 6;
- // First find out how much storage we'll need. Count sections.
- const char* P1 = mP;
- unsigned int est_out = 0;
- while (P1 + 3 < mEnd)
- {
- // read next offset
- uint16_t ofs = *((uint16_t*)P1);
- AI_SWAP2(ofs); P1 += 2;
- if (ofs >= MSZIP_BLOCK)
- throw DeadlyImportError("X: Invalid offset to next MSZIP compressed block");
- // check magic word
- uint16_t magic = *((uint16_t*)P1);
- AI_SWAP2(magic); P1 += 2;
- if (magic != MSZIP_MAGIC)
- throw DeadlyImportError("X: Unsupported compressed format, expected MSZIP header");
- // and advance to the next offset
- P1 += ofs;
- est_out += MSZIP_BLOCK; // one decompressed block is 32786 in size
- }
- // Allocate storage and terminating zero and do the actual uncompressing
- uncompressed.resize(est_out + 1);
- char* out = &uncompressed.front();
- while (mP + 3 < mEnd)
- {
- uint16_t ofs = *((uint16_t*)mP);
- AI_SWAP2(ofs);
- mP += 4;
- if (mP + ofs > mEnd + 2) {
- throw DeadlyImportError("X: Unexpected EOF in compressed chunk");
- }
- // push data to the stream
- stream.next_in = (Bytef*)mP;
- stream.avail_in = ofs;
- stream.next_out = (Bytef*)out;
- stream.avail_out = MSZIP_BLOCK;
- // and decompress the data ....
- int ret = ::inflate( &stream, Z_SYNC_FLUSH );
- if (ret != Z_OK && ret != Z_STREAM_END)
- throw DeadlyImportError("X: Failed to decompress MSZIP-compressed data");
- ::inflateReset( &stream );
- ::inflateSetDictionary( &stream, (const Bytef*)out , MSZIP_BLOCK - stream.avail_out );
- // and advance to the next offset
- out += MSZIP_BLOCK - stream.avail_out;
- mP += ofs;
- }
- // terminate zlib
- ::inflateEnd(&stream);
- // ok, update pointers to point to the uncompressed file data
- mP = &uncompressed[0];
- mEnd = out;
- // FIXME: we don't need the compressed data anymore, could release
- // it already for better memory usage. Consider breaking const-co.
- ASSIMP_LOG_INFO("Successfully decompressed MSZIP-compressed file");
- #endif // !! ASSIMP_BUILD_NO_COMPRESSED_X
- }
- else
- {
- // start reading here
- ReadUntilEndOfLine();
- }
- mScene = new Scene;
- ParseFile();
- // filter the imported hierarchy for some degenerated cases
- if( mScene->mRootNode) {
- FilterHierarchy( mScene->mRootNode);
- }
- }
- // ------------------------------------------------------------------------------------------------
- // Destructor. Destroys all imported data along with it
- XFileParser::~XFileParser()
- {
- // kill everything we created
- delete mScene;
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseFile()
- {
- bool running = true;
- while( running )
- {
- // read name of next object
- std::string objectName = GetNextToken();
- if (objectName.length() == 0)
- break;
- // parse specific object
- if( objectName == "template")
- ParseDataObjectTemplate();
- else
- if( objectName == "Frame")
- ParseDataObjectFrame( NULL);
- else
- if( objectName == "Mesh")
- {
- // some meshes have no frames at all
- Mesh* mesh = new Mesh;
- ParseDataObjectMesh( mesh);
- mScene->mGlobalMeshes.push_back( mesh);
- } else
- if( objectName == "AnimTicksPerSecond")
- ParseDataObjectAnimTicksPerSecond();
- else
- if( objectName == "AnimationSet")
- ParseDataObjectAnimationSet();
- else
- if( objectName == "Material")
- {
- // Material outside of a mesh or node
- Material material;
- ParseDataObjectMaterial( &material);
- mScene->mGlobalMaterials.push_back( material);
- } else
- if( objectName == "}")
- {
- // whatever?
- ASSIMP_LOG_WARN("} found in dataObject");
- } else
- {
- // unknown format
- ASSIMP_LOG_WARN("Unknown data object in animation of .x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectTemplate()
- {
- // parse a template data object. Currently not stored.
- std::string name;
- readHeadOfDataObject( &name);
- // read GUID
- std::string guid = GetNextToken();
- // read and ignore data members
- bool running = true;
- while ( running )
- {
- std::string s = GetNextToken();
- if( s == "}")
- break;
- if( s.length() == 0)
- ThrowException( "Unexpected end of file reached while parsing template definition");
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectFrame( Node* pParent)
- {
- // A coordinate frame, or "frame of reference." The Frame template
- // is open and can contain any object. The Direct3D extensions (D3DX)
- // mesh-loading functions recognize Mesh, FrameTransformMatrix, and
- // Frame template instances as child objects when loading a Frame
- // instance.
- std::string name;
- readHeadOfDataObject(&name);
- // create a named node and place it at its parent, if given
- Node* node = new Node( pParent);
- node->mName = name;
- if( pParent)
- {
- pParent->mChildren.push_back( node);
- } else
- {
- // there might be multiple root nodes
- if( mScene->mRootNode != NULL)
- {
- // place a dummy root if not there
- if( mScene->mRootNode->mName != "$dummy_root")
- {
- Node* exroot = mScene->mRootNode;
- mScene->mRootNode = new Node( NULL);
- mScene->mRootNode->mName = "$dummy_root";
- mScene->mRootNode->mChildren.push_back( exroot);
- exroot->mParent = mScene->mRootNode;
- }
- // put the new node as its child instead
- mScene->mRootNode->mChildren.push_back( node);
- node->mParent = mScene->mRootNode;
- } else
- {
- // it's the first node imported. place it as root
- mScene->mRootNode = node;
- }
- }
- // Now inside a frame.
- // read tokens until closing brace is reached.
- bool running = true;
- while ( running )
- {
- std::string objectName = GetNextToken();
- if (objectName.size() == 0)
- ThrowException( "Unexpected end of file reached while parsing frame");
- if( objectName == "}")
- break; // frame finished
- else
- if( objectName == "Frame")
- ParseDataObjectFrame( node); // child frame
- else
- if( objectName == "FrameTransformMatrix")
- ParseDataObjectTransformationMatrix( node->mTrafoMatrix);
- else
- if( objectName == "Mesh")
- {
- Mesh* mesh = new Mesh(name);
- node->mMeshes.push_back( mesh);
- ParseDataObjectMesh( mesh);
- } else
- {
- ASSIMP_LOG_WARN("Unknown data object in frame in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectTransformationMatrix( aiMatrix4x4& pMatrix)
- {
- // read header, we're not interested if it has a name
- readHeadOfDataObject();
- // read its components
- pMatrix.a1 = ReadFloat(); pMatrix.b1 = ReadFloat();
- pMatrix.c1 = ReadFloat(); pMatrix.d1 = ReadFloat();
- pMatrix.a2 = ReadFloat(); pMatrix.b2 = ReadFloat();
- pMatrix.c2 = ReadFloat(); pMatrix.d2 = ReadFloat();
- pMatrix.a3 = ReadFloat(); pMatrix.b3 = ReadFloat();
- pMatrix.c3 = ReadFloat(); pMatrix.d3 = ReadFloat();
- pMatrix.a4 = ReadFloat(); pMatrix.b4 = ReadFloat();
- pMatrix.c4 = ReadFloat(); pMatrix.d4 = ReadFloat();
- // trailing symbols
- CheckForSemicolon();
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMesh( Mesh* pMesh)
- {
- std::string name;
- readHeadOfDataObject( &name);
- // read vertex count
- unsigned int numVertices = ReadInt();
- pMesh->mPositions.resize( numVertices);
- // read vertices
- for( unsigned int a = 0; a < numVertices; a++)
- pMesh->mPositions[a] = ReadVector3();
- // read position faces
- unsigned int numPosFaces = ReadInt();
- pMesh->mPosFaces.resize( numPosFaces);
- for( unsigned int a = 0; a < numPosFaces; ++a) {
- // read indices
- unsigned int numIndices = ReadInt();
- Face& face = pMesh->mPosFaces[a];
- for (unsigned int b = 0; b < numIndices; ++b) {
- face.mIndices.push_back( ReadInt() );
- }
- TestForSeparator();
- }
- // here, other data objects may follow
- bool running = true;
- while ( running ) {
- std::string objectName = GetNextToken();
- if( objectName.empty() )
- ThrowException( "Unexpected end of file while parsing mesh structure");
- else
- if( objectName == "}")
- break; // mesh finished
- else
- if( objectName == "MeshNormals")
- ParseDataObjectMeshNormals( pMesh);
- else
- if( objectName == "MeshTextureCoords")
- ParseDataObjectMeshTextureCoords( pMesh);
- else
- if( objectName == "MeshVertexColors")
- ParseDataObjectMeshVertexColors( pMesh);
- else
- if( objectName == "MeshMaterialList")
- ParseDataObjectMeshMaterialList( pMesh);
- else
- if( objectName == "VertexDuplicationIndices")
- ParseUnknownDataObject(); // we'll ignore vertex duplication indices
- else
- if( objectName == "XSkinMeshHeader")
- ParseDataObjectSkinMeshHeader( pMesh);
- else
- if( objectName == "SkinWeights")
- ParseDataObjectSkinWeights( pMesh);
- else
- {
- ASSIMP_LOG_WARN("Unknown data object in mesh in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectSkinWeights( Mesh *pMesh) {
- if ( nullptr == pMesh ) {
- return;
- }
- readHeadOfDataObject();
- std::string transformNodeName;
- GetNextTokenAsString( transformNodeName);
- pMesh->mBones.push_back( Bone());
- Bone& bone = pMesh->mBones.back();
- bone.mName = transformNodeName;
- // read vertex weights
- unsigned int numWeights = ReadInt();
- bone.mWeights.reserve( numWeights);
- for( unsigned int a = 0; a < numWeights; a++)
- {
- BoneWeight weight;
- weight.mVertex = ReadInt();
- bone.mWeights.push_back( weight);
- }
- // read vertex weights
- for( unsigned int a = 0; a < numWeights; a++)
- bone.mWeights[a].mWeight = ReadFloat();
- // read matrix offset
- bone.mOffsetMatrix.a1 = ReadFloat(); bone.mOffsetMatrix.b1 = ReadFloat();
- bone.mOffsetMatrix.c1 = ReadFloat(); bone.mOffsetMatrix.d1 = ReadFloat();
- bone.mOffsetMatrix.a2 = ReadFloat(); bone.mOffsetMatrix.b2 = ReadFloat();
- bone.mOffsetMatrix.c2 = ReadFloat(); bone.mOffsetMatrix.d2 = ReadFloat();
- bone.mOffsetMatrix.a3 = ReadFloat(); bone.mOffsetMatrix.b3 = ReadFloat();
- bone.mOffsetMatrix.c3 = ReadFloat(); bone.mOffsetMatrix.d3 = ReadFloat();
- bone.mOffsetMatrix.a4 = ReadFloat(); bone.mOffsetMatrix.b4 = ReadFloat();
- bone.mOffsetMatrix.c4 = ReadFloat(); bone.mOffsetMatrix.d4 = ReadFloat();
- CheckForSemicolon();
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectSkinMeshHeader( Mesh* /*pMesh*/ )
- {
- readHeadOfDataObject();
- /*unsigned int maxSkinWeightsPerVertex =*/ ReadInt();
- /*unsigned int maxSkinWeightsPerFace =*/ ReadInt();
- /*unsigned int numBonesInMesh = */ReadInt();
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMeshNormals( Mesh* pMesh)
- {
- readHeadOfDataObject();
- // read count
- unsigned int numNormals = ReadInt();
- pMesh->mNormals.resize( numNormals);
- // read normal vectors
- for( unsigned int a = 0; a < numNormals; a++)
- pMesh->mNormals[a] = ReadVector3();
- // read normal indices
- unsigned int numFaces = ReadInt();
- if( numFaces != pMesh->mPosFaces.size())
- ThrowException( "Normal face count does not match vertex face count.");
- for( unsigned int a = 0; a < numFaces; a++)
- {
- unsigned int numIndices = ReadInt();
- pMesh->mNormFaces.push_back( Face());
- Face& face = pMesh->mNormFaces.back();
- for( unsigned int b = 0; b < numIndices; b++)
- face.mIndices.push_back( ReadInt());
- TestForSeparator();
- }
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMeshTextureCoords( Mesh* pMesh)
- {
- readHeadOfDataObject();
- if( pMesh->mNumTextures + 1 > AI_MAX_NUMBER_OF_TEXTURECOORDS)
- ThrowException( "Too many sets of texture coordinates");
- std::vector<aiVector2D>& coords = pMesh->mTexCoords[pMesh->mNumTextures++];
- unsigned int numCoords = ReadInt();
- if( numCoords != pMesh->mPositions.size())
- ThrowException( "Texture coord count does not match vertex count");
- coords.resize( numCoords);
- for( unsigned int a = 0; a < numCoords; a++)
- coords[a] = ReadVector2();
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMeshVertexColors( Mesh* pMesh)
- {
- readHeadOfDataObject();
- if( pMesh->mNumColorSets + 1 > AI_MAX_NUMBER_OF_COLOR_SETS)
- ThrowException( "Too many colorsets");
- std::vector<aiColor4D>& colors = pMesh->mColors[pMesh->mNumColorSets++];
- unsigned int numColors = ReadInt();
- if( numColors != pMesh->mPositions.size())
- ThrowException( "Vertex color count does not match vertex count");
- colors.resize( numColors, aiColor4D( 0, 0, 0, 1));
- for( unsigned int a = 0; a < numColors; a++)
- {
- unsigned int index = ReadInt();
- if( index >= pMesh->mPositions.size())
- ThrowException( "Vertex color index out of bounds");
- colors[index] = ReadRGBA();
- // HACK: (thom) Maxon Cinema XPort plugin puts a third separator here, kwxPort puts a comma.
- // Ignore gracefully.
- if( !mIsBinaryFormat)
- {
- FindNextNoneWhiteSpace();
- if( *mP == ';' || *mP == ',')
- mP++;
- }
- }
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMeshMaterialList( Mesh* pMesh)
- {
- readHeadOfDataObject();
- // read material count
- /*unsigned int numMaterials =*/ ReadInt();
- // read non triangulated face material index count
- unsigned int numMatIndices = ReadInt();
- // some models have a material index count of 1... to be able to read them we
- // replicate this single material index on every face
- if( numMatIndices != pMesh->mPosFaces.size() && numMatIndices != 1)
- ThrowException( "Per-Face material index count does not match face count.");
- // read per-face material indices
- for( unsigned int a = 0; a < numMatIndices; a++)
- pMesh->mFaceMaterials.push_back( ReadInt());
- // in version 03.02, the face indices end with two semicolons.
- // commented out version check, as version 03.03 exported from blender also has 2 semicolons
- if( !mIsBinaryFormat) // && MajorVersion == 3 && MinorVersion <= 2)
- {
- if(mP < mEnd && *mP == ';')
- ++mP;
- }
- // if there was only a single material index, replicate it on all faces
- while( pMesh->mFaceMaterials.size() < pMesh->mPosFaces.size())
- pMesh->mFaceMaterials.push_back( pMesh->mFaceMaterials.front());
- // read following data objects
- bool running = true;
- while ( running )
- {
- std::string objectName = GetNextToken();
- if( objectName.size() == 0)
- ThrowException( "Unexpected end of file while parsing mesh material list.");
- else
- if( objectName == "}")
- break; // material list finished
- else
- if( objectName == "{")
- {
- // template materials
- std::string matName = GetNextToken();
- Material material;
- material.mIsReference = true;
- material.mName = matName;
- pMesh->mMaterials.push_back( material);
- CheckForClosingBrace(); // skip }
- } else
- if( objectName == "Material")
- {
- pMesh->mMaterials.push_back( Material());
- ParseDataObjectMaterial( &pMesh->mMaterials.back());
- } else
- if( objectName == ";")
- {
- // ignore
- } else
- {
- ASSIMP_LOG_WARN("Unknown data object in material list in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectMaterial( Material* pMaterial)
- {
- std::string matName;
- readHeadOfDataObject( &matName);
- if( matName.empty())
- matName = std::string( "material") + to_string( mLineNumber );
- pMaterial->mName = matName;
- pMaterial->mIsReference = false;
- // read material values
- pMaterial->mDiffuse = ReadRGBA();
- pMaterial->mSpecularExponent = ReadFloat();
- pMaterial->mSpecular = ReadRGB();
- pMaterial->mEmissive = ReadRGB();
- // read other data objects
- bool running = true;
- while ( running )
- {
- std::string objectName = GetNextToken();
- if( objectName.size() == 0)
- ThrowException( "Unexpected end of file while parsing mesh material");
- else
- if( objectName == "}")
- break; // material finished
- else
- if( objectName == "TextureFilename" || objectName == "TextureFileName")
- {
- // some exporters write "TextureFileName" instead.
- std::string texname;
- ParseDataObjectTextureFilename( texname);
- pMaterial->mTextures.push_back( TexEntry( texname));
- } else
- if( objectName == "NormalmapFilename" || objectName == "NormalmapFileName")
- {
- // one exporter writes out the normal map in a separate filename tag
- std::string texname;
- ParseDataObjectTextureFilename( texname);
- pMaterial->mTextures.push_back( TexEntry( texname, true));
- } else
- {
- ASSIMP_LOG_WARN("Unknown data object in material in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectAnimTicksPerSecond()
- {
- readHeadOfDataObject();
- mScene->mAnimTicksPerSecond = ReadInt();
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectAnimationSet()
- {
- std::string animName;
- readHeadOfDataObject( &animName);
- Animation* anim = new Animation;
- mScene->mAnims.push_back( anim);
- anim->mName = animName;
- bool running = true;
- while ( running )
- {
- std::string objectName = GetNextToken();
- if( objectName.length() == 0)
- ThrowException( "Unexpected end of file while parsing animation set.");
- else
- if( objectName == "}")
- break; // animation set finished
- else
- if( objectName == "Animation")
- ParseDataObjectAnimation( anim);
- else
- {
- ASSIMP_LOG_WARN("Unknown data object in animation set in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectAnimation( Animation* pAnim)
- {
- readHeadOfDataObject();
- AnimBone* banim = new AnimBone;
- pAnim->mAnims.push_back( banim);
- bool running = true;
- while( running )
- {
- std::string objectName = GetNextToken();
- if( objectName.length() == 0)
- ThrowException( "Unexpected end of file while parsing animation.");
- else
- if( objectName == "}")
- break; // animation finished
- else
- if( objectName == "AnimationKey")
- ParseDataObjectAnimationKey( banim);
- else
- if( objectName == "AnimationOptions")
- ParseUnknownDataObject(); // not interested
- else
- if( objectName == "{")
- {
- // read frame name
- banim->mBoneName = GetNextToken();
- CheckForClosingBrace();
- } else
- {
- ASSIMP_LOG_WARN("Unknown data object in animation in x file");
- ParseUnknownDataObject();
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectAnimationKey( AnimBone* pAnimBone)
- {
- readHeadOfDataObject();
- // read key type
- unsigned int keyType = ReadInt();
- // read number of keys
- unsigned int numKeys = ReadInt();
- for( unsigned int a = 0; a < numKeys; a++)
- {
- // read time
- unsigned int time = ReadInt();
- // read keys
- switch( keyType)
- {
- case 0: // rotation quaternion
- {
- // read count
- if( ReadInt() != 4)
- ThrowException( "Invalid number of arguments for quaternion key in animation");
- aiQuatKey key;
- key.mTime = double( time);
- key.mValue.w = ReadFloat();
- key.mValue.x = ReadFloat();
- key.mValue.y = ReadFloat();
- key.mValue.z = ReadFloat();
- pAnimBone->mRotKeys.push_back( key);
- CheckForSemicolon();
- break;
- }
- case 1: // scale vector
- case 2: // position vector
- {
- // read count
- if( ReadInt() != 3)
- ThrowException( "Invalid number of arguments for vector key in animation");
- aiVectorKey key;
- key.mTime = double( time);
- key.mValue = ReadVector3();
- if( keyType == 2)
- pAnimBone->mPosKeys.push_back( key);
- else
- pAnimBone->mScaleKeys.push_back( key);
- break;
- }
- case 3: // combined transformation matrix
- case 4: // denoted both as 3 or as 4
- {
- // read count
- if( ReadInt() != 16)
- ThrowException( "Invalid number of arguments for matrix key in animation");
- // read matrix
- MatrixKey key;
- key.mTime = double( time);
- key.mMatrix.a1 = ReadFloat(); key.mMatrix.b1 = ReadFloat();
- key.mMatrix.c1 = ReadFloat(); key.mMatrix.d1 = ReadFloat();
- key.mMatrix.a2 = ReadFloat(); key.mMatrix.b2 = ReadFloat();
- key.mMatrix.c2 = ReadFloat(); key.mMatrix.d2 = ReadFloat();
- key.mMatrix.a3 = ReadFloat(); key.mMatrix.b3 = ReadFloat();
- key.mMatrix.c3 = ReadFloat(); key.mMatrix.d3 = ReadFloat();
- key.mMatrix.a4 = ReadFloat(); key.mMatrix.b4 = ReadFloat();
- key.mMatrix.c4 = ReadFloat(); key.mMatrix.d4 = ReadFloat();
- pAnimBone->mTrafoKeys.push_back( key);
- CheckForSemicolon();
- break;
- }
- default:
- ThrowException( format() << "Unknown key type " << keyType << " in animation." );
- break;
- } // end switch
- // key separator
- CheckForSeparator();
- }
- CheckForClosingBrace();
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseDataObjectTextureFilename( std::string& pName)
- {
- readHeadOfDataObject();
- GetNextTokenAsString( pName);
- CheckForClosingBrace();
- // FIX: some files (e.g. AnimationTest.x) have "" as texture file name
- if (!pName.length())
- {
- ASSIMP_LOG_WARN("Length of texture file name is zero. Skipping this texture.");
- }
- // some exporters write double backslash paths out. We simply replace them if we find them
- while( pName.find( "\\\\") != std::string::npos)
- pName.replace( pName.find( "\\\\"), 2, "\\");
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ParseUnknownDataObject()
- {
- // find opening delimiter
- bool running = true;
- while( running )
- {
- std::string t = GetNextToken();
- if( t.length() == 0)
- ThrowException( "Unexpected end of file while parsing unknown segment.");
- if( t == "{")
- break;
- }
- unsigned int counter = 1;
- // parse until closing delimiter
- while( counter > 0)
- {
- std::string t = GetNextToken();
- if( t.length() == 0)
- ThrowException( "Unexpected end of file while parsing unknown segment.");
- if( t == "{")
- ++counter;
- else
- if( t == "}")
- --counter;
- }
- }
- // ------------------------------------------------------------------------------------------------
- //! checks for closing curly brace
- void XFileParser::CheckForClosingBrace()
- {
- if( GetNextToken() != "}")
- ThrowException( "Closing brace expected.");
- }
- // ------------------------------------------------------------------------------------------------
- //! checks for one following semicolon
- void XFileParser::CheckForSemicolon()
- {
- if( mIsBinaryFormat)
- return;
- if( GetNextToken() != ";")
- ThrowException( "Semicolon expected.");
- }
- // ------------------------------------------------------------------------------------------------
- //! checks for a separator char, either a ',' or a ';'
- void XFileParser::CheckForSeparator()
- {
- if( mIsBinaryFormat)
- return;
- std::string token = GetNextToken();
- if( token != "," && token != ";")
- ThrowException( "Separator character (';' or ',') expected.");
- }
- // ------------------------------------------------------------------------------------------------
- // tests and possibly consumes a separator char, but does nothing if there was no separator
- void XFileParser::TestForSeparator()
- {
- if( mIsBinaryFormat)
- return;
- FindNextNoneWhiteSpace();
- if( mP >= mEnd)
- return;
- // test and skip
- if( *mP == ';' || *mP == ',')
- mP++;
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::readHeadOfDataObject( std::string* poName)
- {
- std::string nameOrBrace = GetNextToken();
- if( nameOrBrace != "{")
- {
- if( poName)
- *poName = nameOrBrace;
- if ( GetNextToken() != "{" ) {
- delete mScene;
- ThrowException( "Opening brace expected." );
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- std::string XFileParser::GetNextToken() {
- std::string s;
- // process binary-formatted file
- if( mIsBinaryFormat) {
- // in binary mode it will only return NAME and STRING token
- // and (correctly) skip over other tokens.
- if ( mEnd - mP < 2 ) {
- return s;
- }
- unsigned int tok = ReadBinWord();
- unsigned int len;
- // standalone tokens
- switch( tok ) {
- case 1: {
- // name token
- if ( mEnd - mP < 4 ) {
- return s;
- }
- len = ReadBinDWord();
- const int bounds = int( mEnd - mP );
- const int iLen = int( len );
- if ( iLen < 0 ) {
- return s;
- }
- if ( bounds < iLen ) {
- return s;
- }
- s = std::string( mP, len );
- mP += len;
- }
- return s;
- case 2:
- // string token
- if( mEnd - mP < 4) return s;
- len = ReadBinDWord();
- if( mEnd - mP < int(len)) return s;
- s = std::string(mP, len);
- mP += (len + 2);
- return s;
- case 3:
- // integer token
- mP += 4;
- return "<integer>";
- case 5:
- // GUID token
- mP += 16;
- return "<guid>";
- case 6:
- if( mEnd - mP < 4) return s;
- len = ReadBinDWord();
- mP += (len * 4);
- return "<int_list>";
- case 7:
- if( mEnd - mP < 4) return s;
- len = ReadBinDWord();
- mP += (len * mBinaryFloatSize);
- return "<flt_list>";
- case 0x0a:
- return "{";
- case 0x0b:
- return "}";
- case 0x0c:
- return "(";
- case 0x0d:
- return ")";
- case 0x0e:
- return "[";
- case 0x0f:
- return "]";
- case 0x10:
- return "<";
- case 0x11:
- return ">";
- case 0x12:
- return ".";
- case 0x13:
- return ",";
- case 0x14:
- return ";";
- case 0x1f:
- return "template";
- case 0x28:
- return "WORD";
- case 0x29:
- return "DWORD";
- case 0x2a:
- return "FLOAT";
- case 0x2b:
- return "DOUBLE";
- case 0x2c:
- return "CHAR";
- case 0x2d:
- return "UCHAR";
- case 0x2e:
- return "SWORD";
- case 0x2f:
- return "SDWORD";
- case 0x30:
- return "void";
- case 0x31:
- return "string";
- case 0x32:
- return "unicode";
- case 0x33:
- return "cstring";
- case 0x34:
- return "array";
- }
- }
- // process text-formatted file
- else
- {
- FindNextNoneWhiteSpace();
- if( mP >= mEnd)
- return s;
- while( (mP < mEnd) && !isspace( (unsigned char) *mP))
- {
- // either keep token delimiters when already holding a token, or return if first valid char
- if( *mP == ';' || *mP == '}' || *mP == '{' || *mP == ',')
- {
- if( !s.size())
- s.append( mP++, 1);
- break; // stop for delimiter
- }
- s.append( mP++, 1);
- }
- }
- return s;
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::FindNextNoneWhiteSpace()
- {
- if( mIsBinaryFormat)
- return;
- bool running = true;
- while( running )
- {
- while( mP < mEnd && isspace( (unsigned char) *mP))
- {
- if( *mP == '\n')
- mLineNumber++;
- ++mP;
- }
- if( mP >= mEnd)
- return;
- // check if this is a comment
- if( (mP[0] == '/' && mP[1] == '/') || mP[0] == '#')
- ReadUntilEndOfLine();
- else
- break;
- }
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::GetNextTokenAsString( std::string& poString)
- {
- if( mIsBinaryFormat)
- {
- poString = GetNextToken();
- return;
- }
- FindNextNoneWhiteSpace();
- if ( mP >= mEnd ) {
- delete mScene;
- ThrowException( "Unexpected end of file while parsing string" );
- }
- if ( *mP != '"' ) {
- delete mScene;
- ThrowException( "Expected quotation mark." );
- }
- ++mP;
- while( mP < mEnd && *mP != '"')
- poString.append( mP++, 1);
- if ( mP >= mEnd - 1 ) {
- delete mScene;
- ThrowException( "Unexpected end of file while parsing string" );
- }
- if ( mP[ 1 ] != ';' || mP[ 0 ] != '"' ) {
- delete mScene;
- ThrowException( "Expected quotation mark and semicolon at the end of a string." );
- }
- mP+=2;
- }
- // ------------------------------------------------------------------------------------------------
- void XFileParser::ReadUntilEndOfLine()
- {
- if( mIsBinaryFormat)
- return;
- while( mP < mEnd)
- {
- if( *mP == '\n' || *mP == '\r')
- {
- ++mP; mLineNumber++;
- return;
- }
- ++mP;
- }
- }
- // ------------------------------------------------------------------------------------------------
- unsigned short XFileParser::ReadBinWord()
- {
- ai_assert(mEnd - mP >= 2);
- const unsigned char* q = (const unsigned char*) mP;
- unsigned short tmp = q[0] | (q[1] << 8);
- mP += 2;
- return tmp;
- }
- // ------------------------------------------------------------------------------------------------
- unsigned int XFileParser::ReadBinDWord() {
- ai_assert(mEnd - mP >= 4);
- const unsigned char* q = (const unsigned char*) mP;
- unsigned int tmp = q[0] | (q[1] << 8) | (q[2] << 16) | (q[3] << 24);
- mP += 4;
- return tmp;
- }
- // ------------------------------------------------------------------------------------------------
- unsigned int XFileParser::ReadInt()
- {
- if( mIsBinaryFormat)
- {
- if( mBinaryNumCount == 0 && mEnd - mP >= 2)
- {
- unsigned short tmp = ReadBinWord(); // 0x06 or 0x03
- if( tmp == 0x06 && mEnd - mP >= 4) // array of ints follows
- mBinaryNumCount = ReadBinDWord();
- else // single int follows
- mBinaryNumCount = 1;
- }
- --mBinaryNumCount;
- if ( mEnd - mP >= 4) {
- return ReadBinDWord();
- } else {
- mP = mEnd;
- return 0;
- }
- } else
- {
- FindNextNoneWhiteSpace();
- // TODO: consider using strtol10 instead???
- // check preceding minus sign
- bool isNegative = false;
- if( *mP == '-')
- {
- isNegative = true;
- mP++;
- }
- // at least one digit expected
- if( !isdigit( *mP))
- ThrowException( "Number expected.");
- // read digits
- unsigned int number = 0;
- while( mP < mEnd)
- {
- if( !isdigit( *mP))
- break;
- number = number * 10 + (*mP - 48);
- mP++;
- }
- CheckForSeparator();
- return isNegative ? ((unsigned int) -int( number)) : number;
- }
- }
- // ------------------------------------------------------------------------------------------------
- ai_real XFileParser::ReadFloat()
- {
- if( mIsBinaryFormat)
- {
- if( mBinaryNumCount == 0 && mEnd - mP >= 2)
- {
- unsigned short tmp = ReadBinWord(); // 0x07 or 0x42
- if( tmp == 0x07 && mEnd - mP >= 4) // array of floats following
- mBinaryNumCount = ReadBinDWord();
- else // single float following
- mBinaryNumCount = 1;
- }
- --mBinaryNumCount;
- if( mBinaryFloatSize == 8) {
- if( mEnd - mP >= 8) {
- double res;
- ::memcpy( &res, mP, 8 );
- mP += 8;
- const ai_real result( static_cast<ai_real>( res ) );
- return result;
- } else {
- mP = mEnd;
- return 0;
- }
- } else {
- if( mEnd - mP >= 4) {
- ai_real result;
- ::memcpy( &result, mP, 4 );
- mP += 4;
- return result;
- } else {
- mP = mEnd;
- return 0;
- }
- }
- }
- // text version
- FindNextNoneWhiteSpace();
- // check for various special strings to allow reading files from faulty exporters
- // I mean you, Blender!
- // Reading is safe because of the terminating zero
- if( strncmp( mP, "-1.#IND00", 9) == 0 || strncmp( mP, "1.#IND00", 8) == 0)
- {
- mP += 9;
- CheckForSeparator();
- return 0.0;
- } else
- if( strncmp( mP, "1.#QNAN0", 8) == 0)
- {
- mP += 8;
- CheckForSeparator();
- return 0.0;
- }
- ai_real result = 0.0;
- mP = fast_atoreal_move<ai_real>( mP, result);
- CheckForSeparator();
- return result;
- }
- // ------------------------------------------------------------------------------------------------
- aiVector2D XFileParser::ReadVector2()
- {
- aiVector2D vector;
- vector.x = ReadFloat();
- vector.y = ReadFloat();
- TestForSeparator();
- return vector;
- }
- // ------------------------------------------------------------------------------------------------
- aiVector3D XFileParser::ReadVector3()
- {
- aiVector3D vector;
- vector.x = ReadFloat();
- vector.y = ReadFloat();
- vector.z = ReadFloat();
- TestForSeparator();
- return vector;
- }
- // ------------------------------------------------------------------------------------------------
- aiColor4D XFileParser::ReadRGBA()
- {
- aiColor4D color;
- color.r = ReadFloat();
- color.g = ReadFloat();
- color.b = ReadFloat();
- color.a = ReadFloat();
- TestForSeparator();
- return color;
- }
- // ------------------------------------------------------------------------------------------------
- aiColor3D XFileParser::ReadRGB()
- {
- aiColor3D color;
- color.r = ReadFloat();
- color.g = ReadFloat();
- color.b = ReadFloat();
- TestForSeparator();
- return color;
- }
- // ------------------------------------------------------------------------------------------------
- // Throws an exception with a line number and the given text.
- AI_WONT_RETURN void XFileParser::ThrowException( const std::string& pText) {
- if ( mIsBinaryFormat ) {
- throw DeadlyImportError( pText );
- } else {
- throw DeadlyImportError( format() << "Line " << mLineNumber << ": " << pText );
- }
- }
- // ------------------------------------------------------------------------------------------------
- // Filters the imported hierarchy for some degenerated cases that some exporters produce.
- void XFileParser::FilterHierarchy( XFile::Node* pNode)
- {
- // if the node has just a single unnamed child containing a mesh, remove
- // the anonymous node between. The 3DSMax kwXport plugin seems to produce this
- // mess in some cases
- if( pNode->mChildren.size() == 1 && pNode->mMeshes.empty() )
- {
- XFile::Node* child = pNode->mChildren.front();
- if( child->mName.length() == 0 && child->mMeshes.size() > 0)
- {
- // transfer its meshes to us
- for( unsigned int a = 0; a < child->mMeshes.size(); a++)
- pNode->mMeshes.push_back( child->mMeshes[a]);
- child->mMeshes.clear();
- // transfer the transform as well
- pNode->mTrafoMatrix = pNode->mTrafoMatrix * child->mTrafoMatrix;
- // then kill it
- delete child;
- pNode->mChildren.clear();
- }
- }
- // recurse
- for( unsigned int a = 0; a < pNode->mChildren.size(); a++)
- FilterHierarchy( pNode->mChildren[a]);
- }
- #endif // !! ASSIMP_BUILD_NO_X_IMPORTER
|