// CheckIfCondition //------------------------------------------------------------------------------ bool BFFParser::CheckIfCondition( const BFFIterator & conditionStart, const BFFIterator & conditionEnd, bool & result ) { // trim condition AStackString<> condition( conditionStart.GetCurrent(), conditionEnd.GetCurrent() ); result = BFFMacros::Get().IsDefined( condition ); return true; }
// ParseNamedVariableName //------------------------------------------------------------------------------ /*static*/ bool BFFParser::ParseVariableName( BFFIterator & iter, AString & name, bool & parentScope ) { // skip over the declaration symbol ASSERT( *iter == BFF_DECLARE_VAR_INTERNAL || *iter == BFF_DECLARE_VAR_PARENT ); parentScope = ( *iter == BFF_DECLARE_VAR_PARENT ); const BFFIterator varNameStart = iter; // include type token in var name iter++; // make sure we haven't hit the end of the file if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } // make sure immediately after the symbol starts a variable name if ( iter.IsAtValidVariableNameCharacter() == false ) { Error::Error_1013_UnexpectedCharInVariableName( iter, nullptr ); return false; } // find the end of the variable name iter.SkipVariableName(); const BFFIterator varNameEnd = iter; // sanity check it is a sensible length size_t varNameLen = varNameStart.GetDistTo( varNameEnd ); if ( varNameLen > MAX_VARIABLE_NAME_LENGTH ) { Error::Error_1014_VariableNameIsTooLong( iter, (uint32_t)varNameLen, (uint32_t)MAX_VARIABLE_NAME_LENGTH ); return false; } // store variable name name.Assign( varNameStart.GetCurrent(), varNameEnd.GetCurrent() ); if ( parentScope ) { // exchange '^' with '.' ASSERT( BFF_DECLARE_VAR_PARENT == name[0] ); name[0] = BFF_DECLARE_VAR_INTERNAL; } return true; }
// ParseDefineDirective //------------------------------------------------------------------------------ bool BFFParser::ParseDefineDirective( BFFIterator & iter ) { if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } // parse out token const BFFIterator tokenStart( iter ); iter.SkipVariableName(); if ( tokenStart.GetCurrent() == iter.GetCurrent() ) { Error::Error_1007_ExpectedVariable( iter, nullptr ); return false; } const BFFIterator tokenEnd( iter ); AStackString<> token( tokenStart.GetCurrent(), tokenEnd.GetCurrent() ); if ( BFFMacros::Get().Define( token ) == false ) { Error::Error_1038_OverwritingTokenInDefine( tokenStart ); return false; } FLOG_INFO( "Define macro <%s>", token.Get() ); return true; }
// ParseUnnamedScope //------------------------------------------------------------------------------ bool BFFParser::ParseUnnamedScope( BFFIterator & iter ) { // find the matching bracket BFFIterator scopeStart( iter ); if ( iter.ParseToMatchingBrace( BFF_SCOPE_OPEN, BFF_SCOPE_CLOSE ) == false ) { Error::Error_1025_MissingScopeCloseToken( scopeStart, nullptr ); return false; } // create stack for scope BFFStackFrame stackFrame; // parse the scoped part BFFParser subParser; BFFIterator subIter( scopeStart ); subIter++; // skip opening token subIter.SetMax( iter.GetCurrent() ); // limit to closing token if ( subParser.Parse( subIter ) == false ) { return false; } iter++; // skip closing } return true; }
// StoreVariableStruct //------------------------------------------------------------------------------ bool BFFParser::StoreVariableStruct( const AString & name, const BFFIterator & valueStart, const BFFIterator & valueEnd, const BFFIterator & operatorIter, BFFStackFrame * frame ) { // are we concatenating? if ( *operatorIter == BFF_VARIABLE_CONCATENATION ) { // concatenation of structs not supported Error::Error_1027_CannotModify( operatorIter, name, BFFVariable::VAR_STRUCT, BFFVariable::VAR_ANY ); return false; } // create stack frame to capture variables BFFStackFrame stackFrame; // parse all the variables in the scope BFFParser subParser; BFFIterator subIter( valueStart ); subIter.SetMax( valueEnd.GetCurrent() ); // limit to closing token if ( subParser.Parse( subIter ) == false ) { return false; // error will be emitted by Parse } // get variables defined in the scope const Array< const BFFVariable * > & structMembers = stackFrame.GetLocalVariables(); // Register this variable BFFStackFrame::SetVarStruct( name, structMembers, frame ? frame : stackFrame.GetParent() ); FLOG_INFO( "Registered <struct> variable '%s' with %u members", name.Get(), structMembers.GetSize() ); return true; }
// CheckIfCondition //------------------------------------------------------------------------------ bool BFFParser::CheckIfCondition( const BFFIterator & conditionStart, const BFFIterator & conditionEnd, bool & result ) { // trim condition AStackString<> condition( conditionStart.GetCurrent(), conditionEnd.GetCurrent() ); condition.Replace( '\t', ' ' ); condition.Replace( " ", "" ); result = false; // For now we only support trivial pre-defined expressions - TODO:B Support more complex expressions if ( condition == "__WINDOWS__" ) { #if defined( __WINDOWS__ ) result = true; #endif return true; } if ( condition == "__LINUX__" ) { #if defined( __LINUX__ ) result = true; #endif return true; } if ( condition == "__OSX__" ) { #if defined( __OSX__ ) result = true; #endif return true; } // We found an expression we don't understand Error::Error_1036_UnknownTokenInIfDirective( conditionStart ); return false; }
// ParseUndefDirective //------------------------------------------------------------------------------ bool BFFParser::ParseUndefDirective( BFFIterator & iter ) { if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } // parse out token const BFFIterator tokenStart( iter ); iter.SkipVariableName(); if ( tokenStart.GetCurrent() == iter.GetCurrent() ) { Error::Error_1007_ExpectedVariable( iter, nullptr ); return false; } const BFFIterator tokenEnd( iter ); AStackString<> token( tokenStart.GetCurrent(), tokenEnd.GetCurrent() ); if ( BFFMacros::Get().Undefine( token ) == false ) { if ( token.BeginsWith( "__" ) ) { Error::Error_1040_UndefOfBuiltInTokenNotAllowed( tokenStart ); } else { Error::Error_1039_UnknownTokenInUndef( tokenStart ); } return false; } FLOG_INFO( "Undefine macro <%s>", token.Get() ); return true; }
// ParseIfDirective //------------------------------------------------------------------------------ bool BFFParser::ParseIfDirective( const BFFIterator & directiveStart, BFFIterator & iter ) { if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } bool negate = false; if ( *iter == '!' ) { negate = true; // the condition will be inverted iter++; // skip '!' iter.SkipWhiteSpace(); // allow whitepace after '!' if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } } // parse out condition const BFFIterator conditionStart( iter ); iter.SkipVariableName(); if ( conditionStart.GetCurrent() == iter.GetCurrent() ) { Error::Error_1007_ExpectedVariable( directiveStart, nullptr ); return false; } const BFFIterator conditionEnd( iter ); // Evaluate the condition bool result; if ( CheckIfCondition( conditionStart, conditionEnd, result ) == false ) { return false; // CheckIfCondition will have emitted an error } // #ifndef ? if ( negate ) { result = !( result ); } if ( result ) { ++s_IfDepth; // Track that we're inside an if block return true; // continue parsing like normal } // Advance iterator past entire #if block size_t depth = 1; // handle nested ifs while ( depth > 0 ) { // did we hit the end of the file? if ( iter.IsAtEnd() ) { (void)directiveStart; // TODO: Show we're looking for matching endif to this Error::Error_1012_UnexpectedEndOfFile( iter ); // TODO:B better error for this? return false; } // find the next preprocessor directive iter.SkipWhiteSpace(); if ( *iter == BFF_PREPROCESSOR_START ) { iter++; // skip # iter.SkipWhiteSpace(); // allow whitespace between # and directive const BFFIterator directiveNameStart( iter ); while ( iter.IsAtValidDirectiveNameCharacter() ) { iter++; } const BFFIterator directiveNameEnd( iter ); AStackString<> directiveName( directiveNameStart.GetCurrent(), directiveNameEnd.GetCurrent() ); if ( directiveName == "endif" ) { --depth; } else if ( directiveName == "if" ) { ++depth; } // continue to skip rest of line.... } // skip rest of line while ( ( iter.IsAtEnd() == false ) && ( *iter != '\r' ) && ( *iter != '\n' ) ) { iter++; } } return true; }
// ParseFunction //------------------------------------------------------------------------------ bool BFFParser::ParseFunction( BFFIterator & iter ) { ASSERT( iter.IsAtValidFunctionNameCharacter() ); // for variables to be used by this function BFFStackFrame stackFrame; BFFIterator functionNameStart( iter ); iter.SkipFunctionName(); if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } // check length if ( functionNameStart.GetDistTo( iter ) > MAX_FUNCTION_NAME_LENGTH ) { // if it's too long, then it can't be a valid function Error::Error_1015_UnknownFunction( functionNameStart ); return false; } // store function name AStackString<MAX_FUNCTION_NAME_LENGTH> functionName( functionNameStart.GetCurrent(), iter.GetCurrent() ); const Function * func = Function::Find( functionName ); if ( func == nullptr ) { Error::Error_1015_UnknownFunction( functionNameStart ); return false; } iter.SkipWhiteSpace(); if ( func->IsUnique() && func->GetSeen() ) { Error::Error_1020_FunctionCanOnlyBeInvokedOnce( functionNameStart, func ); return false; } func->SetSeen(); FLOG_INFO( "Function call '%s'", functionName.Get() ); // header, or body? bool hasHeader = false; BFFIterator functionArgsStartToken( iter ); BFFIterator functionArgsStopToken( iter ); if ( *iter == BFF_FUNCTION_ARGS_OPEN ) { // can this function accept a header? if ( func->AcceptsHeader() == false ) { Error::Error_1021_UnexpectedHeaderForFunction( iter, func ); return false; } // args if ( iter.ParseToMatchingBrace( BFF_FUNCTION_ARGS_OPEN, BFF_FUNCTION_ARGS_CLOSE ) == false ) { Error::Error_1022_MissingFunctionHeaderCloseToken( functionArgsStartToken, func ); return false; } functionArgsStopToken = iter; hasHeader = true; iter++; // skip over closing token iter.SkipWhiteSpaceAndComments(); } if ( func->NeedsHeader() && ( hasHeader == false ) ) { Error::Error_1023_FunctionRequiresAHeader( iter, func ); return false; } // some functions have no body bool hasBody = false; BFFIterator functionBodyStartToken( iter ); BFFIterator functionBodyStopToken( iter ); if ( func->NeedsBody() ) { // find body if ( *iter != BFF_SCOPE_OPEN ) { Error::Error_1024_FunctionRequiresABody( functionNameStart, func ); return false; } if ( iter.ParseToMatchingBrace( BFF_SCOPE_OPEN, BFF_SCOPE_CLOSE ) == false ) { Error::Error_1025_MissingScopeCloseToken( functionBodyStartToken, func ); return false; } functionBodyStopToken = iter; iter++; hasBody = true; } return func->ParseFunction( functionNameStart, hasBody ? &functionBodyStartToken : nullptr, hasBody ? &functionBodyStopToken : nullptr, hasHeader ? &functionArgsStartToken : nullptr, hasHeader ? &functionArgsStopToken : nullptr );}
// ParseVariableDeclaration //------------------------------------------------------------------------------ bool BFFParser::ParseVariableDeclaration( BFFIterator & iter, const AString & varName, BFFStackFrame * frame ) { m_SeenAVariable = true; // look for an appropriate operator BFFIterator operatorIter( iter ); bool modification = false; if ( *iter == BFF_VARIABLE_ASSIGNMENT ) { // assignment } else if ( ( *iter == BFF_VARIABLE_CONCATENATION ) || ( *iter == BFF_VARIABLE_SUBTRACTION ) ) { // concatenation modification = true; } else { Error::Error_1016_UnexepectedCharFollowingVariableName( iter ); return false; } // skip the assignment symbol and whitespace iter++; iter.SkipWhiteSpaceAndComments(); if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } char openToken = *iter; char closeToken = 0; bool ok = false; if ( ( openToken == '"' ) || ( openToken == '\'' ) ) { closeToken = openToken; ok = true; } else if ( openToken == BFF_SCOPE_OPEN ) { closeToken = BFF_SCOPE_CLOSE; ok = true; } else if ( openToken == BFF_STRUCT_OPEN ) { closeToken = BFF_STRUCT_CLOSE; ok = true; } else if ( ( openToken >= '0' ) && ( openToken <= '9' ) ) { if ( modification ) { Error::Error_1027_CannotModify( operatorIter, varName, BFFVariable::VAR_ANY, BFFVariable::VAR_INT ); return false; } // integer value? BFFIterator startIntValue( iter ); while ( iter.IsAtEnd() == false ) { iter++; if ( ( *iter < '0' ) || ( *iter > '9' ) ) { break; // end of integer } } if ( startIntValue.GetDistTo( iter ) > 10 ) { Error::Error_1018_IntegerValueCouldNotBeParsed( startIntValue ); return false; } AStackString<> intAsString( startIntValue.GetCurrent(), iter.GetCurrent() ); int i = 0; if ( sscanf( intAsString.Get(), "%i", &i ) != 1 ) { Error::Error_1018_IntegerValueCouldNotBeParsed( startIntValue ); return false; } return StoreVariableInt( varName, i, frame ); } else if ( ( *iter == 't' ) || ( *iter == 'f' ) ) { // might be 'true' or 'false' BFFIterator startBoolValue( iter ); if ( iter.ParseToNext( 'e' ) == true ) { iter++; if ( ( startBoolValue.GetDistTo( iter ) <= 5 ) ) { AStackString<8> value( startBoolValue.GetCurrent(), iter.GetCurrent() ); if ( value == "true" ) { if ( modification ) { Error::Error_1027_CannotModify( operatorIter, varName, BFFVariable::VAR_ANY, BFFVariable::VAR_BOOL ); return false; } return StoreVariableBool( varName, true, frame ); } else if ( value == "false" ) { if ( modification ) { Error::Error_1027_CannotModify( operatorIter, varName, BFFVariable::VAR_ANY, BFFVariable::VAR_BOOL ); return false; } return StoreVariableBool( varName, false, frame ); } } } // not a valid bool value } else if ( *iter == BFF_DECLARE_VAR_INTERNAL || *iter == BFF_DECLARE_VAR_PARENT ) { return StoreVariableToVariable( varName, iter, operatorIter, frame ); } if ( !ok ) { Error::Error_1017_UnexepectedCharInVariableValue( iter ); return false; } bool result = false; // find closing token BFFIterator openTokenPos( iter ); BFFIterator openTokenForError( iter ); // take note of opening token pos for possible error openTokenPos++; // more start iter to first char of value if ( openToken == BFF_SCOPE_OPEN ) { if ( iter.ParseToMatchingBrace( openToken, closeToken ) ) { result = StoreVariableArray( varName, openTokenPos, iter, operatorIter, frame ); } else { Error::Error_1002_MatchingClosingTokenNotFound( openTokenForError, nullptr, closeToken ); } } else if ( openToken == BFF_STRUCT_OPEN ) { if ( iter.ParseToMatchingBrace( openToken, closeToken ) ) { result = StoreVariableStruct( varName, openTokenPos, iter, operatorIter, frame ); } else { Error::Error_1002_MatchingClosingTokenNotFound( openTokenForError, nullptr, closeToken ); } } else { ASSERT( ( openToken == '\'' ) || ( openToken == '"' ) ); iter.SkipString( closeToken ); if ( *iter == closeToken ) { result = StoreVariableString( varName, openTokenPos, iter, operatorIter, frame ); } else { Error::Error_1002_MatchingClosingTokenNotFound( openTokenForError, nullptr, closeToken ); } } if ( result ) { iter++; // skip over the end token } // StoreVariable will have emitted an error if there was one return result; }
// ParseNamedVariableName //------------------------------------------------------------------------------ /*static*/ bool BFFParser::ParseVariableName( BFFIterator & iter, AString & name, bool & parentScope ) { // skip over the declaration symbol ASSERT( *iter == BFF_DECLARE_VAR_INTERNAL || *iter == BFF_DECLARE_VAR_PARENT ); parentScope = ( *iter == BFF_DECLARE_VAR_PARENT ); const BFFIterator varNameStart = iter; // include type token in var name iter++; // make sure we haven't hit the end of the file if ( iter.IsAtEnd() ) { Error::Error_1012_UnexpectedEndOfFile( iter ); return false; } if ( *iter == '\'' || *iter == '"' ) { // parse the string const BFFIterator openToken = iter; iter.SkipString( *openToken ); if ( *iter != *openToken ) { Error::Error_1002_MatchingClosingTokenNotFound( openToken, nullptr, *openToken ); return false; } BFFIterator stringStart = openToken; stringStart++; // unescape and subsitute embedded variables AStackString< 256 > value; if ( PerformVariableSubstitutions( stringStart, iter, value ) == false ) { return false; } iter++; // skip close token BFFIterator varNameIter( value.Get(), value.GetLength(), iter.GetFileName().Get(), iter.GetFileTimeStamp() ); // sanity check it is a sensible length if ( value.GetLength() + 1/* '.' will be added */ > MAX_VARIABLE_NAME_LENGTH ) { Error::Error_1014_VariableNameIsTooLong( varNameIter, (uint32_t)value.GetLength(), (uint32_t)MAX_VARIABLE_NAME_LENGTH ); return false; } // sanity check it is a valid variable name while ( varNameIter.IsAtEnd() == false ) { if ( varNameIter.IsAtValidVariableNameCharacter() == false ) { Error::Error_1013_UnexpectedCharInVariableName( varNameIter, nullptr ); return false; } varNameIter++; } // append '.' to variable name name = "."; name.Append( value ); } else { // make sure immediately after the symbol starts a variable name if ( iter.IsAtValidVariableNameCharacter() == false ) { Error::Error_1013_UnexpectedCharInVariableName( iter, nullptr ); return false; } // find the end of the variable name iter.SkipVariableName(); const BFFIterator varNameEnd = iter; // sanity check it is a sensible length size_t varNameLen = varNameStart.GetDistTo( varNameEnd ); if ( varNameLen > MAX_VARIABLE_NAME_LENGTH ) { Error::Error_1014_VariableNameIsTooLong( iter, (uint32_t)varNameLen, (uint32_t)MAX_VARIABLE_NAME_LENGTH ); return false; } // store variable name name.Assign( varNameStart.GetCurrent(), varNameEnd.GetCurrent() ); } ASSERT( name.GetLength() > 0 ); if ( parentScope ) { // exchange '^' with '.' ASSERT( BFF_DECLARE_VAR_PARENT == name[0] ); name[0] = BFF_DECLARE_VAR_INTERNAL; } return true; }
// StoreVariableArray //------------------------------------------------------------------------------ bool BFFParser::StoreVariableArray( const AString & name, const BFFIterator & valueStart, const BFFIterator & valueEnd, const BFFIterator & operatorIter, BFFStackFrame * frame ) { Array< AString > values( 32, true ); Array< const BFFVariable * > structValues( 32, true ); // are we concatenating? if ( ( *operatorIter == BFF_VARIABLE_CONCATENATION ) || ( *operatorIter == BFF_VARIABLE_SUBTRACTION ) ) { // find existing const BFFVariable * var = BFFStackFrame::GetVar( name, frame ); if ( var == nullptr ) { Error::Error_1026_VariableNotFoundForModification( operatorIter, name ); return false; } // make sure existing is an array if ( var->IsArrayOfStrings() ) { // get values to start with values = var->GetArrayOfStrings(); } else if ( var->IsArrayOfStructs() ) { // get values to start with structValues = var->GetArrayOfStructs(); } else { // TODO:B Improve this error to handle ArrayOfStructs case Error::Error_1027_CannotModify( operatorIter, name, var->GetType(), BFFVariable::VAR_ARRAY_OF_STRINGS ); return false; } } // Parse array of variables BFFIterator iter( valueStart ); for (;;) { iter.SkipWhiteSpaceAndComments(); // end? if ( iter.GetCurrent() == valueEnd.GetCurrent() ) { break; } const char c = *iter; if ( ( c == '"' ) || ( c == '\'' ) ) { // a quoted string // detect mistmatches if ( structValues.IsEmpty() == false ) { // Mixed types in vector Error::Error_1034_OperationNotSupported( iter, BFFVariable::VAR_ARRAY_OF_STRUCTS, BFFVariable::VAR_STRING, operatorIter ); return false; } // subtraction not supported on arrays if ( *operatorIter == BFF_VARIABLE_SUBTRACTION ) { Error::Error_1034_OperationNotSupported( iter, BFFVariable::VAR_ARRAY_OF_STRINGS, BFFVariable::VAR_STRING, operatorIter ); return false; } // a string BFFIterator elementValueStart( iter ); iter.SkipString( c ); ASSERT( iter.GetCurrent() <= valueEnd.GetCurrent() ); // should not be in this function if string is not terminated elementValueStart++; // move to start of actual content AStackString< 2048 > elementValue; // unescape and subsitute embedded variables if ( PerformVariableSubstitutions( elementValueStart, iter, elementValue ) == false ) { return false; } values.Append( elementValue ); iter++; // pass closing quote } else if ( c == BFF_DECLARE_VAR_INTERNAL || c == BFF_DECLARE_VAR_PARENT ) { const BFFIterator elementStartValue = iter; // a variable AStackString< MAX_VARIABLE_NAME_LENGTH > varName; bool parentScope = false; // ignored, the behavior is the same if ( ParseVariableName( iter, varName, parentScope ) == false ) { return false; } // Determine stack frame to use for Src var BFFStackFrame * srcFrame = BFFStackFrame::GetCurrent(); if ( c == BFF_DECLARE_VAR_PARENT ) { srcFrame = BFFStackFrame::GetCurrent()->GetParent(); } // get the variable const BFFVariable * var = srcFrame->GetVariableRecurse( varName ); if ( var == nullptr ) { Error::Error_1026_VariableNotFoundForModification( operatorIter, varName ); return false; } // subtraction not supported on arrays if ( *operatorIter == BFF_VARIABLE_SUBTRACTION ) { const BFFVariable::VarType dstType = structValues.IsEmpty() ? BFFVariable::VAR_ARRAY_OF_STRINGS : BFFVariable::VAR_ARRAY_OF_STRUCTS; const BFFVariable::VarType srcType = var->GetType(); Error::Error_1034_OperationNotSupported( elementStartValue, dstType, srcType, operatorIter ); return false; } if ( var->IsString() || var->IsArrayOfStrings() ) { // dest is consistent? if ( structValues.IsEmpty() == false ) { // inconsistency Error::Error_1034_OperationNotSupported( elementStartValue, BFFVariable::VAR_ARRAY_OF_STRUCTS, var->GetType(), operatorIter ); return false; } if ( var->IsString() ) { values.Append( var->GetString() ); } else { values.Append( var->GetArrayOfStrings() ); } } else if ( var->IsStruct() || var->IsArrayOfStructs() ) { // dest is consistent? if ( values.IsEmpty() == false ) { // inconsistency Error::Error_1034_OperationNotSupported( elementStartValue, BFFVariable::VAR_ARRAY_OF_STRINGS, var->GetType(), operatorIter ); return false; } if ( var->IsStruct() ) { structValues.Append( var ); } else { structValues.Append( var->GetArrayOfStructs() ); } } else { Error::Error_1050_PropertyMustBeOfType( iter, nullptr, name.Get(), var->GetType(), BFFVariable::VAR_STRING, BFFVariable::VAR_STRUCT ); return false; } } else { Error::Error_1001_MissingStringStartToken( iter, nullptr ); return false; } iter.SkipWhiteSpaceAndComments(); if ( *iter == ',' ) // comma seperators are optional { iter++; } // continue looking for more vars... } // should only have one populated array ASSERT( values.IsEmpty() || structValues.IsEmpty() ); // Register this variable if ( structValues.IsEmpty() == false ) { // structs BFFStackFrame::SetVarArrayOfStructs( name, structValues, frame ); FLOG_INFO( "Registered <ArrayOfStructs> variable '%s' with %u elements", name.Get(), structValues.GetSize() ); } else { // strings (or unknown, so default to strings) BFFStackFrame::SetVarArrayOfStrings( name, values, frame ); FLOG_INFO( "Registered <ArrayOfStrings> variable '%s' with %u elements", name.Get(), values.GetSize() ); } return true; }
// //------------------------------------------------------------------------------ bool BFFParser::StoreVariableToVariable( const char * varNameDstStart, const char * varNameDstEnd, const BFFIterator & varNameSrcStart, const BFFIterator & varNameSrcEnd, const BFFIterator & operatorIter ) { // find vars AStackString<> dstName( varNameDstStart, varNameDstEnd ); AStackString<> srcName( varNameSrcStart.GetCurrent(), varNameSrcEnd.GetCurrent() ); const BFFVariable * varDst = BFFStackFrame::GetVar( dstName ); const BFFVariable * varSrc = BFFStackFrame::GetVar( srcName ); // src var unknown? if ( varSrc == nullptr ) { Error::Error_1009_UnknownVariable( varNameSrcStart, nullptr ); return false; } const bool concat = ( *operatorIter == BFF_VARIABLE_CONCATENATION ); // concatenation? if ( concat ) { // can only concatenate to existing vars if ( varDst == nullptr ) { Error::Error_1026_VariableNotFoundForConcatenation( operatorIter, dstName ); return false; } } // if dst exists, types must match BFFVariable::VarType srcType = varSrc->GetType(); BFFVariable::VarType dstType = BFFVariable::VAR_ANY; if ( varDst ) { dstType = varDst->GetType(); } else { ASSERT( concat == false ); dstType = srcType; } // handle supported types if ( srcType != dstType ) { // Mismatched - is there a supported conversion? // String to ArrayOfStrings if ( ( dstType == BFFVariable::VAR_ARRAY_OF_STRINGS ) && ( srcType == BFFVariable::VAR_STRING ) ) { uint32_t num = (uint32_t)( 1 + ( concat ? varDst->GetArrayOfStrings().GetSize() : 0 ) ); Array< AString > values( num, false ); if ( concat ) { values.Append( varDst->GetArrayOfStrings() ); } values.Append( varSrc->GetString() ); BFFStackFrame::SetVarArrayOfStrings( dstName, values ); FLOG_INFO( "Registered <ArrayOfStrings> variable '%s' with %u elements", dstName.Get(), num ); return true; } // Struct to ArrayOfStructs if ( ( dstType == BFFVariable::VAR_ARRAY_OF_STRUCTS ) && ( srcType == BFFVariable::VAR_STRUCT ) ) { uint32_t num = (uint32_t)( 1 + ( concat ? varDst->GetArrayOfStructs().GetSize() : 0 ) ); Array< const BFFVariable * > values( num, false ); if ( concat ) { values.Append( varDst->GetArrayOfStructs() ); } values.Append( varSrc ); BFFStackFrame::SetVarArrayOfStructs( dstName, values ); FLOG_INFO( "Registered <ArrayOfStructs> variable '%s' with %u elements", dstName.Get(), num ); return true; } } else { // Matching Src and Dst if ( srcType == BFFVariable::VAR_STRING ) { AStackString< 2048 > finalValue; if ( concat ) { finalValue = varDst->GetString(); } finalValue += varSrc->GetString(); BFFStackFrame::SetVarString( dstName, finalValue ); FLOG_INFO( "Registered <string> variable '%s' with value '%s'", dstName.Get(), finalValue.Get() ); return true; } if ( srcType == BFFVariable::VAR_ARRAY_OF_STRINGS ) { if ( concat ) { const unsigned int num = (unsigned int)( varSrc->GetArrayOfStrings().GetSize() + varDst->GetArrayOfStrings().GetSize() ); Array< AString > values( num, false ); values.Append( varDst->GetArrayOfStrings() ); values.Append( varSrc->GetArrayOfStrings() ); BFFStackFrame::SetVarArrayOfStrings( dstName, values ); FLOG_INFO( "Registered <ArrayOfStrings> variable '%s' with %u elements", dstName.Get(), num ); } else { BFFStackFrame::SetVarArrayOfStrings( dstName, varSrc->GetArrayOfStrings() ); FLOG_INFO( "Registered <ArrayOfStrings> variable '%s' with %u elements", dstName.Get(), (unsigned int)varSrc->GetArrayOfStrings().GetSize() ); } return true; } if ( srcType == BFFVariable::VAR_ARRAY_OF_STRUCTS ) { if ( concat ) { const unsigned int num = (unsigned int)( varSrc->GetArrayOfStructs().GetSize() + varDst->GetArrayOfStructs().GetSize() ); Array< const BFFVariable * > values( num, false ); values.Append( varDst->GetArrayOfStructs() ); values.Append( varSrc->GetArrayOfStructs() ); BFFStackFrame::SetVarArrayOfStructs( dstName, values ); FLOG_INFO( "Registered <ArrayOfStructs> variable '%s' with %u elements", dstName.Get(), num ); } else { BFFStackFrame::SetVarArrayOfStructs( dstName, varSrc->GetArrayOfStructs() ); FLOG_INFO( "Registered <ArrayOfStructs> variable '%s' with %u elements", dstName.Get(), (unsigned int)varSrc->GetArrayOfStructs().GetSize() ); } return true; } if ( srcType == BFFVariable::VAR_INT ) { int newVal( varSrc->GetInt() ); if ( concat ) { newVal += varDst->GetInt(); } return StoreVariableInt( varNameDstStart, varNameDstEnd, newVal ); } if ( srcType == BFFVariable::VAR_BOOL ) { // only assignment is supported if ( concat == false ) { return StoreVariableBool( varNameDstStart, varNameDstEnd, varSrc->GetBool() ); } } if ( srcType == BFFVariable::VAR_STRUCT ) { const Array< const BFFVariable * > & srcMembers = varSrc->GetStructMembers(); if ( concat ) { BFFVariable *const newVar = BFFStackFrame::ConcatVars( dstName, varSrc, varDst ); FLOG_INFO( "Registered <struct> variable '%s' with %u members", dstName.Get(), newVar->GetStructMembers().GetSize() ); } else { // Register this variable BFFStackFrame::SetVarStruct( dstName, srcMembers ); FLOG_INFO( "Registered <struct> variable '%s' with %u members", dstName.Get(), srcMembers.GetSize() ); } return true; } } Error::Error_1034_OperationNotSupported( varNameSrcStart, varDst ? varDst->GetType() : varSrc->GetType(), varSrc->GetType(), operatorIter ); return false; }