123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344 |
- /*
- Open Asset Import Library (assimp)
- ----------------------------------------------------------------------
- Copyright (c) 2006-2012, 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 FBXParser.cpp
- * @brief Implementation of the FBX parser and the rudimentary DOM that we use
- */
- #include "AssimpPCH.h"
- #ifndef ASSIMP_BUILD_NO_FBX_IMPORTER
- #include "FBXTokenizer.h"
- #include "FBXParser.h"
- #include "FBXUtil.h"
- #include "ParsingUtils.h"
- #include "fast_atof.h"
- using namespace Assimp;
- using namespace Assimp::FBX;
- namespace {
- // ------------------------------------------------------------------------------------------------
- // signal parsing error, this is always unrecoverable. Throws DeadlyImportError.
- void ParseError(const std::string& message, TokenPtr token)
- {
- throw DeadlyImportError(token ? Util::AddTokenText("FBX-Parse",message,token) : ("FBX-Parse " + message));
- }
- }
- namespace Assimp {
- namespace FBX {
- // ------------------------------------------------------------------------------------------------
- Element::Element(const Token& key_token, Parser& parser)
- : key_token(key_token)
- {
- TokenPtr n = NULL;
- do {
- n = parser.AdvanceToNextToken();
- if(!n) {
- ParseError("unexpected end of file, expected closing bracket",parser.LastToken());
- }
- if (n->Type() == TokenType_DATA) {
- tokens.push_back(n);
- n = parser.AdvanceToNextToken();
- if(!n) {
- ParseError("unexpected end of file, expected bracket, comma or key",parser.LastToken());
- }
- const TokenType ty = n->Type();
- if (ty != TokenType_OPEN_BRACKET && ty != TokenType_CLOSE_BRACKET && ty != TokenType_COMMA && ty != TokenType_KEY) {
- ParseError("unexpected token; expected bracket, comma or key",n);
- }
- }
- if (n->Type() == TokenType_OPEN_BRACKET) {
- compound.reset(new Scope(parser));
- // current token should be a TOK_CLOSE_BRACKET
- n = parser.CurrentToken();
- ai_assert(n);
- if (n->Type() != TokenType_CLOSE_BRACKET) {
- ParseError("expected closing bracket",n);
- }
- parser.AdvanceToNextToken();
- return;
- }
- }
- while(n->Type() != TokenType_KEY && n->Type() != TokenType_CLOSE_BRACKET);
- }
- // ------------------------------------------------------------------------------------------------
- Element::~Element()
- {
- // no need to delete tokens, they are owned by the parser
- }
- // ------------------------------------------------------------------------------------------------
- Scope::Scope(Parser& parser,bool topLevel)
- {
- if(!topLevel) {
- TokenPtr t = parser.CurrentToken();
- if (t->Type() != TokenType_OPEN_BRACKET) {
- ParseError("expected open bracket",t);
- }
- }
- TokenPtr n = parser.AdvanceToNextToken();
- if(n == NULL) {
- ParseError("unexpected end of file",NULL);
- }
- // note: empty scopes are allowed
- while(n->Type() != TokenType_CLOSE_BRACKET) {
- if (n->Type() != TokenType_KEY) {
- ParseError("unexpected token, expected TOK_KEY",n);
- }
- const std::string& str = n->StringContents();
- elements.insert(ElementMap::value_type(str,new_Element(*n,parser)));
- // Element() should stop at the next Key token (or right after a Close token)
- n = parser.CurrentToken();
- if(n == NULL) {
- if (topLevel) {
- return;
- }
- ParseError("unexpected end of file",parser.LastToken());
- }
- }
- }
- // ------------------------------------------------------------------------------------------------
- Scope::~Scope()
- {
- BOOST_FOREACH(ElementMap::value_type& v, elements) {
- delete v.second;
- }
- }
- // ------------------------------------------------------------------------------------------------
- Parser::Parser (const TokenList& tokens)
- : tokens(tokens)
- , cursor(tokens.begin())
- , current()
- , last()
- {
- root.reset(new Scope(*this,true));
- }
- // ------------------------------------------------------------------------------------------------
- Parser::~Parser()
- {
- }
- // ------------------------------------------------------------------------------------------------
- TokenPtr Parser::AdvanceToNextToken()
- {
- last = current;
- if (cursor == tokens.end()) {
- current = NULL;
- }
- else {
- current = *cursor++;
- }
- return current;
- }
- // ------------------------------------------------------------------------------------------------
- TokenPtr Parser::CurrentToken() const
- {
- return current;
- }
- // ------------------------------------------------------------------------------------------------
- TokenPtr Parser::LastToken() const
- {
- return last;
- }
- // ------------------------------------------------------------------------------------------------
- uint64_t ParseTokenAsID(const Token& t, const char*& err_out)
- {
- err_out = NULL;
- if (t.Type() != TokenType_DATA) {
- err_out = "expected TOK_DATA token";
- return 0L;
- }
- // XXX: should use size_t here
- unsigned int length = static_cast<unsigned int>(t.end() - t.begin());
- ai_assert(length > 0);
- const char* out;
- const uint64_t id = strtoul10_64(t.begin(),&out,&length);
- if (out != t.end()) {
- err_out = "failed to parse ID";
- return 0L;
- }
- return id;
- }
- // ------------------------------------------------------------------------------------------------
- uint64_t ParseTokenAsDim(const Token& t, const char*& err_out)
- {
- // same as ID parsing, except there is a trailing asterisk
- err_out = NULL;
- if (t.Type() != TokenType_DATA) {
- err_out = "expected TOK_DATA token";
- return 0L;
- }
- if(*t.begin() != '*') {
- err_out = "expected asterisk before array dimension";
- return 0L;
- }
- // XXX: should use size_t here
- unsigned int length = static_cast<unsigned int>(t.end() - t.begin());
- if(length == 0) {
- err_out = "expected valid integer number after asterisk";
- return 0L;
- }
- const char* out;
- const uint64_t id = strtoul10_64(t.begin() + 1,&out,&length);
- if (out != t.end()) {
- err_out = "failed to parse ID";
- return 0L;
- }
- return id;
- }
- // ------------------------------------------------------------------------------------------------
- float ParseTokenAsFloat(const Token& t, const char*& err_out)
- {
- err_out = NULL;
- if (t.Type() != TokenType_DATA) {
- err_out = "expected TOK_DATA token";
- return 0.0f;
- }
- const char* inout = t.begin();
- float f;
- fast_atof(&inout);
- if (inout != t.end()) {
- err_out = "failed to parse floating point number";
- return 0.0f;
- }
- return f;
- }
- // ------------------------------------------------------------------------------------------------
- int ParseTokenAsInt(const Token& t, const char*& err_out)
- {
- err_out = NULL;
- if (t.Type() != TokenType_DATA) {
- err_out = "expected TOK_DATA token";
- return 0;
- }
- ai_assert(static_cast<size_t>(t.end() - t.begin()) > 0);
- const char* out;
- const int intval = strtol10(t.begin(),&out);
- if (out != t.end()) {
- err_out = "failed to parse ID";
- return 0;
- }
- return intval;
- }
- // ------------------------------------------------------------------------------------------------
- std::string ParseTokenAsString(const Token& t, const char*& err_out)
- {
- err_out = NULL;
- if (t.Type() != TokenType_DATA) {
- err_out = "expected TOK_DATA token";
- return "";
- }
- const size_t length = static_cast<size_t>(t.end() - t.begin());
- if(length < 2) {
- err_out = "token is too short to hold a string";
- return "";
- }
- const char* s = t.begin(), *e = t.end() - 1;
- if (*s != '\"' || *e != '\*') {
- err_out = "expected double quoted string";
- return "";
- }
- return std::string(s+1,length-2);
- }
- } // !FBX
- } // !Assimp
- #endif
|