diff -r 820b22e13ff1 -r 39c28ec933dd bintools/rcomp/src/NUMVAL.CPP --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/bintools/rcomp/src/NUMVAL.CPP Mon May 10 19:54:49 2010 +0100 @@ -0,0 +1,473 @@ +/* +* Copyright (c) 1997-2009 Nokia Corporation and/or its subsidiary(-ies). +* All rights reserved. +* This component and the accompanying materials are made available +* under the terms of the License "Eclipse Public License v1.0" +* which accompanies this distribution, and is available +* at the URL "http://www.eclipse.org/legal/epl-v10.html". +* +* Initial Contributors: +* Nokia Corporation - initial contribution. +* +* Contributors: +* +* Description: +* +*/ + + +#include <assert.h> +#include <string.h> +#include <ctype.h> +#include <stdlib.h> + +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__) +#include <sstream> +#include <iostream> +using std::cerr; +using std::endl; +#else //!__MSVCDOTNET__ +#ifndef __LINUX__ +#include <strstrea.h> +#endif //!__LINUX__ +#endif //__MSVCDOTNET__ + +#include "ASTRING.H" +#include "NUMVAL.H" +#include "STRUCTST.H" +#include "Parser.h" +#include "rcomp.hpp" +#include "MEM.H" +#include "ERRORHAN.H" +#include "RCBINSTR.H" + +#if defined(__VC32__) +#pragma warning( disable : 4702 ) // unreachable code +#endif + +NumericValue::NumericValue( const String & Source, DataType NumericValueType): + iNumericValueType( NumericValueType), + iData( NULL), + iULongValue( 0), + iSignedValue( 0), + iDoubleValue( 0.0) + { + AllocateSpace(); + ConvertToNumber( Source); + } + +NumericValue::NumericValue( DataType NumericValueType): + iNumericValueType( NumericValueType), + iData( NULL), + iULongValue( 0), + iSignedValue( 0), + iDoubleValue( 0.0) + { + AllocateSpace(); + } + +NumericValue::NumericValue( unsigned long ValueToSet, DataType NumericValueType): + iNumericValueType( NumericValueType), + iData( NULL), + iULongValue( 0), + iSignedValue( 0), + iDoubleValue( 0.0) + { + AllocateSpace(); + StoreValue( ValueToSet); + } + +NumericValue::~NumericValue() + { + delete [] iData; + } + +NumericValue::NumericValue( const NumericValue & Source): + iNumericValueType( Source.iNumericValueType), + iData( NULL), + iULongValue( 0), + iSignedValue( 0), + iDoubleValue( 0.0) + { + AllocateSpace(); + memcpy( iData, Source.iData, iSize); + } + +void NumericValue::AllocateSpace() + { + switch(iNumericValueType) + { + case L_BYTE: + iData = new unsigned char [1]; + iSize = 1; + break; + case L_WORD: + iData = new unsigned char [2]; + iSize = 2; + break; + case L_LONG: + iData = new unsigned char [4]; + iSize = 4; + break; + default: + if ( iNumericValueType != L_DOUBLE) + assert(0); // Cannot use NumericValue for specified data type. + } + + if ( iNumericValueType != L_DOUBLE && iData == NULL) + { + ErrorHandler::OutputErrorLine( "Failed to allocate space for number."); + exit(1); + } + } + +const unsigned char * NumericValue::Data() const + { + return iData; + } + +unsigned long NumericValue::Size() const + { + return iSize; + } + +DataType NumericValue::NumericValueType() const + { + return iNumericValueType; + } + +void NumericValue::ConvertToNumber( const String & Source) + { + if ( iNumericValueType == L_DOUBLE) + ConvertToDouble( Source); + else + ConvertToNatural( Source); + } + +void NumericValue::ConvertToDouble( const String & Source) + { + assert( iNumericValueType == L_DOUBLE); + assert( Source.Length() > 0); + + double d = atof( Source.GetAssertedNonEmptyBuffer()); + if ( d == 0.0 && !( Source == "0.0" || Source == "0") ) + { MOFF; cerr << "atof may have failed for " << Source << endl; MON;} + + iDoubleValue = d; + } + +#if defined(__VC32__) +#pragma warning( disable : 4706 ) // assignment within conditional expression +#endif + +void NumericValue::ConvertToNatural( const String & Source) + { + unsigned long LongValue = 0; + + assert( sizeof( unsigned long) >= 4); // Assume that LongValue can hold at least 2^32 - 1. + + const char * pSourceChar = Source.iRep; + int bLeadingHyphen = 0; + int bHexNumber = 0; + + if ( pSourceChar[0] == '0' && pSourceChar[1] == 'x') + { + bHexNumber = 1; + pSourceChar++; + pSourceChar++; + } + + if ( pSourceChar[0] == '-') + { + bLeadingHyphen = 1; + pSourceChar++; + } + + while ( * pSourceChar != '\0') + { + unsigned char DigitValue; + + if ( bHexNumber) + { + assert( isxdigit( * pSourceChar) ); + if ( isdigit( * pSourceChar) ) + DigitValue = (unsigned char)(* pSourceChar - '0'); + else + DigitValue = (unsigned char)(toupper( * pSourceChar) - 'A' + 10); + if (LongValue >= 0x10000000) + { + String st("Number \""); + st += Source; + st += "\" is too big "; + ErrorHandler::OutputErrorLine(st); //prevents overflow if number is bigger than 2^32 - 1. + } + LongValue = LongValue * 16 + DigitValue; + } + else + { + if ( ! isdigit( * pSourceChar) ) + { + String s( "Cannot convert \""); + s += Source; + s += "\" to a number."; + ErrorHandler::OutputErrorLine( s); + exit(1); + // unreachable code + } + DigitValue = (unsigned char)(* pSourceChar - '0'); + if ((LongValue > 429496729) || ((LongValue == 429496729) && (DigitValue > 5))) + { + String st("Number \""); + st += Source; + st += "\" is too big "; + ErrorHandler::OutputErrorLine(st); //prevents overflow if number is bigger than 2^32 - 1. + } + LongValue = LongValue * 10 + DigitValue; + } + pSourceChar++; + assert( ( pSourceChar - Source.iRep) < 10000); // Safety check! + } + + int inrange=0; + + // Check value is within the allowed range for the type taking into account + // a leading hyphen (minus sign) if there was one. + switch( iNumericValueType) + { + case L_BYTE: // L_BYTE is 8 bits long. + if ( bLeadingHyphen) + { + if ( ! ( inrange = (LongValue <= 128) ) ) // 2 ^ 7 + ErrorHandler::OutputErrorLine( "Number too low for BYTE"); + } + else + if ( ! ( inrange = (LongValue <= 0xFF) ) ) + ErrorHandler::OutputErrorLine( "Number too big for BYTE"); + break; + case L_WORD: // L_WORD is 16-bits long. + if ( bLeadingHyphen) + { + if ( ! ( inrange = (LongValue <= 32768) ) ) // 2^15 + ErrorHandler::OutputErrorLine( "Number too low for WORD"); + } + else + if ( ! ( inrange = (LongValue <= 0xFFFF) ) ) + ErrorHandler::OutputErrorLine( "Number too big for WORD"); + break; + case L_LONG: // L_LONG is 32-bits long + if ( bLeadingHyphen) + { + if ( ! ( inrange = (LongValue <= 0x80000000) ) ) // 2^31 + ErrorHandler::OutputErrorLine( "Number too low for LONG"); + } + else + if ( ! ( inrange = (LongValue <= 0xFFFFFFFF ) ) ) // This test is a bit pointless as long cannot be greater than 0xffffffff + ErrorHandler::OutputErrorLine( "Number too big for LONG"); + break; + default: + assert(0); // Other data types cannot be converted to natural numbers. + } + + if(!inrange) + exit(1); + + StoreValue( LongValue); + + // If there was a hyphen then make the stored number negative (using two's complement). + if ( bLeadingHyphen) + { + LongValue = (LongValue ^ 0xFFFFFFFFu)+1; + + // Output file will be treated as little-endian. + switch ( iNumericValueType) + { + case L_LONG: + iData[3] = (unsigned char)((LongValue & 0xFF000000) >> 24); + iData[2] = (unsigned char)((LongValue & 0xFF0000) >> 16); + case L_WORD: + iData[1] = (unsigned char)((LongValue & 0xFF00) >> 8); + case L_BYTE: + iData[0] = (unsigned char)(LongValue & 0xFF); + } + } + } + +#if defined(__VC32__) +#pragma warning( default : 4706 ) // assignment within conditional expression +#endif + +void NumericValue::StoreValue( unsigned long LongValue) + { + iULongValue = LongValue; + + if ( LongValue <= 0x80000000) + iSignedValue = (unsigned long) LongValue; + + int inrange = 1; + + switch( iNumericValueType) + { + case L_BYTE: + inrange = ( LongValue <= 0xFF); + break; + case L_WORD: + inrange = ( LongValue <= 0xFFFF); + break; + case L_LONG: + inrange = ( LongValue <= 0xFFFFFFFF); + } + + if ( ! inrange) + { + ErrorHandler::OutputErrorLine( "Numeric value out of range for specified type"); + exit(1); + } + + // Output file will be treated as little-endian. + switch ( iNumericValueType) + { + case L_LONG: + iData[3] = (unsigned char)((LongValue & 0xFF000000) >> 24); + iData[2] = (unsigned char)((LongValue & 0xFF0000) >> 16); + case L_WORD: + iData[1] = (unsigned char)((LongValue & 0xFF00) >> 8); + case L_BYTE: + iData[0] = (unsigned char)(LongValue & 0xFF); + } + } + +template<bool> class __CompileTimeAssert {public: __CompileTimeAssert(...) {}}; +template<> class __CompileTimeAssert<false> {}; +struct COMPILE_TIME_ERROR {}; +#define COMPILE_TIME_ASSERT(aCondition) { __CompileTimeAssert<(aCondition)> __temp = __CompileTimeAssert<(aCondition)>(COMPILE_TIME_ERROR()); } + +RCBinaryStream & operator<< ( RCBinaryStream & os, NumericValue o) + { + switch( o.iNumericValueType) + { + case L_BYTE: + os.Write( o.iData, 1); + break; + case L_WORD: + os.Write( o.iData, 2); + break; + case L_LONG: + os.Write( o.iData, 4); + break; + case L_DOUBLE: + COMPILE_TIME_ASSERT(sizeof(double) == 8); + os.Write(reinterpret_cast<unsigned char*>(&o.iDoubleValue), 8); + break; + default: + assert(0); + } + + return os; + } + +void NumericValue::StreamOut(ResourceDataStream& aStream) const + { + switch (iNumericValueType) + { + case L_BYTE: + aStream.StreamIn(iData, 1); + break; + case L_WORD: + aStream.StreamIn(iData, 2); + break; + case L_LONG: + aStream.StreamIn(iData, 4); + break; + case L_DOUBLE: + COMPILE_TIME_ASSERT(sizeof(double) == 8); + aStream.StreamIn(reinterpret_cast<const unsigned char*>(&iDoubleValue), 8); + break; + default: + assert(0); + } + } + +NumericValue & NumericValue::operator= ( unsigned long ValueToSet) + { + StoreValue( ValueToSet); + + return * this; + } + +unsigned long NumericValue::GetULong() const + { + return iULongValue; + } + +long NumericValue::GetLong() const + { + assert( iULongValue <= 2147483647); // Check that we are not holding a number that is really positive only. + return iSignedValue; + } + +bool NumericValue::CheckSigned(unsigned long aValue, DataType aNumericValueType) + { + switch (aNumericValueType) + { + case L_BYTE: + if (aValue > 0x7f) return false; + break; + case L_WORD: + if (aValue > 0x7fff) return false; + break; + case L_LONG: + if (aValue > 0x7fffffff) return false; + break; + default: + assert(0); + } + return true; + } + +String NumericValue::ltoa( long Source) + { + char v[10]; // long can have no more than 10 digits in this implementation. + char * pv = v; + long x; + + if ( Source < 0) + x = - Source; + else + x = Source; + + if ( x == 0) + * pv++ = '0'; + else + { + while( x > 0) + { + assert( pv <= (v+9) ); + + * pv = char(x%10 + '0'); + pv++; + x /= 10; + } + } + + // Now reverse digits so they are in the correct order. Put in terminating null and hyphen + // if necessary. + + char r[12]; + char * pr = r; + + if ( Source < 0) + { + r[0] = '-'; + pr++; + } + + while( pv != v) + { + assert( pr < (r+11) ); + * pr++ = * --pv; + } + + * pr = '\0'; + + return r; + }