diff -r 820b22e13ff1 -r 39c28ec933dd bintools/rcomp/src/RCOSTRM.CPP --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/bintools/rcomp/src/RCOSTRM.CPP Mon May 10 19:54:49 2010 +0100 @@ -0,0 +1,542 @@ +/* +* 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 +#include +#include + +#include "RESOURCE.H" +#include "DATATYPE.H" +#include "Parser.h" +#include "rcomp.hpp" +#include "NUMVAL.H" +#include "MEM.H" +#include "ERRORHAN.H" +#include "UNICODE_COMPRESSOR.H" +#include "main.h" + +#if defined(__MSVCDOTNET__) || defined(__TOOLS2__) +using std::endl; +using std::cout; +#endif //__MSVCDOTNET__ + +extern int verbose; +extern String::CharacterSet SourceCharacterSet; + +void Panic(int aCode) // used by UNICODE_COMPRESSOR.CPP + { + exit(aCode); + } + +RCTypeArray gTypes; + +// StructItem based ostream functions + +ostream& SimpleStructItem::StreamOut ( ostream & os) + { + return operator<< ( os, * this); + } + +ostream& ArrayStructItem::StreamOut ( ostream & os) + { + return operator<< ( os, * this); + } + +ostream& StructTypeStructItem::StreamOut ( ostream & os) + { + return operator<< ( os, * this); + } + +ostream& StructArrayStructItem::StreamOut ( ostream & os) + { + return operator<< ( os, * this); + } + +ostream& operator<< ( ostream & os, SimpleStructItem & o) + { + os << "SimpleStructItem "; + os << o.iLabel; + os << "\t" << gTypes.GetName( o.iItemType); + os << "\tDefault: " << o.iDefault; + os << "\tLength limit: " << o.iLengthLimit; + os << endl; + return os ; + } + +ostream& operator<< ( ostream & os, ArrayStructItem & o) + { + os << "ArrayStructItem "; + os << o.iLabel; + os << "\t" << gTypes.GetName( o.iItemType); + os << "\tDefaults: " << o.iDefaults; + os << "\tLen Type: " << ((o.iLenType == 0) ? "" : gTypes.GetName(o.iLenType)); + os << "\tSize: " << o.iSize; + os << endl; + return os ; + } + +ostream& operator<< ( ostream & os, StructTypeStructItem & o) + { + os << "StructTypeStructItem "; + os << o.iLabel; + os << endl; + return os; + } + +ostream& operator<< ( ostream & os, StructArrayStructItem & o) + { + os << "StructArrayStructItem "; + os << o.iLabel; + os << "\tLen Type: " << ( ( o.iLenType == 0) ? "" : gTypes.GetName( o.iLenType) ); + os << "\tSize: " << o.iSize; + os << endl; + return os; + } + +ostream& operator<< ( ostream & os, StructItemArray & o) + { + if (o.Size() == 0) + return ( os << ""); + StructItemArrayIterator next(o); + StructItem* p; + while ((p = next() ) != NULL) + p->StreamOut(os); + return os; + } + +ostream& operator<< ( ostream & os, StructHeader & o) + { + os << "StructHeader "; + os << o.iLabel << endl; + os << "\tLen Type: " << ( ( o.iLenType == 0) ? "" : gTypes.GetName( o.iLenType) ) << endl; + os << o.iSIA; + return os; + } + +ostream& operator<< ( ostream & os, StructHeaderArray & o) + { + if (o.Size() == 0) + return ( os << ""); + StructHeaderArrayIterator next( o); + StructHeader* p; + while ( ( p = next() ) != NULL) + os << *p; + return os; + } + +// ResourceItem based ostream functions + +ostream& SimpleResourceItem::StreamOut(ostream& os) + { + return operator<<(os,*this); + } + +ostream& ArrayResourceItem::StreamOut(ostream& os) + { + return operator<<(os,*this); + } + +ostream& StructTypeResourceItem::StreamOut(ostream& os) + { + return operator<<(os,*this); + } + +ostream& StructArrayResourceItem::StreamOut(ostream& os) + { + return operator<<(os,*this); + } + +ostream & operator<<(ostream& os,SimpleResourceItem& o) + { + os << "SimpleResourceItem [" << gTypes.GetName(o.iStructItem->iItemType) << " " << o.iStructItem->iLabel << "] "; + switch( o.iStructItem->iItemType) + { + case L_TEXT: + case L_LTEXT: + case L_BUF: + case L_BYTE: + case L_WORD: + case L_LONG: + os << o.iValue; + break; + case L_SRLINK: + break; // Don't know value yet. + } + os << endl; + return os; + } + +ostream & operator<< ( ostream & os, ArrayResourceItem & o) + { + os << "ArrayResourceItem [" << o.iStructItem->iLabel << "] "; + os << o.iValues; + os << endl; + return os; + } + +ostream & operator<< ( ostream & os, StructTypeResourceItem & o) + { + static unsigned long level = 0; // Recursion level. + os << "StructTypeResourceItem (Level " << level++ << ") [" << o.iStructItem->iLabel << "] " << endl; + os << "----------------------" << endl; + os << o.iResourceItems; + os << "----------------------" << endl; + level--; + return os; + } + +ostream & operator<< ( ostream & os, StructArrayResourceItem & o) + { + static unsigned long level = 0; // Recursion level. + os << "StructArrayResourceItem (Level " << level++ << ") [" << o.iStructItem->iLabel << "] " << endl; + DataType counttype = o.iStructItem->iLenType; + if (counttype==0) + counttype = L_WORD; + unsigned long nitems = o.iArrayOfResourceItemArrays.Size(); + os << "ArrayLength [" << gTypes.GetName(counttype) << "] " << nitems << endl; + if (nitems > 0) + { + os << "----------------------" << endl; + os << o.iArrayOfResourceItemArrays; + os << "----------------------" << endl; + } + level--; + return os; + } + +ostream & operator<< ( ostream & os, ResourceHeader & o) + { + os << "ResourceHeader "; + os << o.iLabel << endl; + os << o.iRIA; + return os; + } + +ostream & operator<< ( ostream & os, ResourceItemArray & o) + { + os << "++ResourceItemArray" << endl; + if ( o.iLenType != 0) + { + os << "LenType: " << gTypes.GetName( o.iLenType) << "\t"; + } + if ( o.Size() == 0) + return ( os << ""); + ResourceItemArrayIterator next( o); + ResourceItem * p; + while ( ( p = next() ) != NULL) + p->StreamOut( os); + return os; + } + +ostream & operator<< ( ostream & os, ResourceItemArrayArray & o) + { + os << "--ResourceItemArrayArray" << endl; + if ( o.Size() == 0) + return ( os << ""); + ResourceItemArrayArrayIterator next( o); + ResourceItemArray * p; + while ( ( p = next() ) != NULL) + os << * p; + return os; + } + +void StreamOutCompressingIfReducesSize(ResourceDataStream& aStream, const UTF16* aUncompressedUnicodeBuffer, int aUncompressedUnicodeLength) + { + const int uncompressedUnicodeSizeInBytes = aUncompressedUnicodeLength*2; + + // + // Output buffer allocated to be one byte bigger than uncompressed input buffer. + // This is to allow for what appears to be the worst case where the string cannot be + // compressed at all. In this case enough memory has to be allocated to contain + // the uncompressable string, plus the one-byte compression mode tag. + // + const int maximumOutputLengthInBytes = uncompressedUnicodeSizeInBytes; + unsigned char* const compressedUnicodeBuffer = new unsigned char[maximumOutputLengthInBytes]; + assert(compressedUnicodeBuffer != NULL); + + int compressedUnicodeSizeInBytes; + CompressUnicode(compressedUnicodeBuffer, compressedUnicodeSizeInBytes, maximumOutputLengthInBytes, aUncompressedUnicodeBuffer, aUncompressedUnicodeLength); + + if (compressedUnicodeSizeInBytes < uncompressedUnicodeSizeInBytes) + { + aStream.StartOfCompressedUnicodeRun(uncompressedUnicodeSizeInBytes, (const unsigned char*)aUncompressedUnicodeBuffer); + aStream.StreamIn(compressedUnicodeBuffer, compressedUnicodeSizeInBytes); + aStream.EndOfCompressedUnicodeRun(); + } + else if (uncompressedUnicodeSizeInBytes>0) + { + aStream.TwoByteAlignmentPoint(); + aStream.StreamIn((const unsigned char*)aUncompressedUnicodeBuffer, uncompressedUnicodeSizeInBytes); + } + + delete [] compressedUnicodeBuffer; + } + +// ResourceItem based RCBinaryStream functions +// These are used to write the raw data as required in the final output file + +static void StreamOutSingleValue( + ResourceDataStream& aStream, + const String & aValue, + unsigned long aLinkValue, + DataType aItemType, + const String * aFileName, + int aLineNumber + ) + { + ErrorHandler::Register( aFileName, aLineNumber); // Register in case conversion to number fails. + + + + switch ( aItemType) + { + + // 8-bit text + + case L_TEXT8: + { // Null terminator, no leading byte count. + unsigned char * p = new unsigned char[ aValue.Length() + 1]; + if ( aValue.Length() > 0) + memcpy( p, aValue.GetAssertedNonEmptyBuffer(), aValue.Length() ); + * ( p + aValue.Length() ) = '\0'; + aStream.StreamIn(p, aValue.Length() + 1); + delete [] p; + break; + } + case L_LTEXT8: + { // Leading byte count, no null terminator. (For zero length string a byte containing + // zero is written out. + assert ( aValue.Length() <= 255 ); + unsigned char * p = new unsigned char[ aValue.Length() + 1]; + if ( aValue.Length() > 0) + memcpy( p + 1, aValue.GetAssertedNonEmptyBuffer(), aValue.Length() ); + *p = (unsigned char)(aValue.Length()); + aStream.StreamIn(p, aValue.Length() + 1); + delete [] p; + break; + } + case L_BUF8: // Write just the characters from the string. + aStream.StreamIn((const unsigned char*)aValue.GetAssertedNonEmptyBuffer(), aValue.Length()); + break; + + + // 16-bit text + + + + case L_BUF16: // write out unadorned 16-bit characters + { + int output_count; // used as character count for binary output. + + // reserve enough for worst case + output_count = aValue.Length(); + UTF16 *output_buffer = new UTF16[output_count]; + + aValue.Export(output_buffer, output_count, SourceCharacterSet); + + // write out the buffer as a stream of octets. + #if defined(__TOOLS2_WINDOWS__) + StreamOutCompressingIfReducesSize(aStream, (const unsigned short*)output_buffer, output_count); + #else + StreamOutCompressingIfReducesSize(aStream, output_buffer, output_count); + #endif + + delete [] output_buffer; + + break; + } + + + + case L_TEXT16: // write out a null-terminated string + // this has no support in BAFL + { + int output_count; // used as character count for binary output. + + // reserve enough for worst case + null terminator. + output_count = aValue.Length(); + UTF16 *output_buffer = new UTF16[output_count +1]; + + aValue.Export(output_buffer, output_count, SourceCharacterSet); + output_buffer[output_count] = 0; + output_count +=1; + + // write out the buffer as a stream of octets. + #if defined(__TOOLS2_WINDOWS__) + StreamOutCompressingIfReducesSize(aStream, (const unsigned short*)output_buffer, output_count); + #else + StreamOutCompressingIfReducesSize(aStream, output_buffer, output_count); + #endif + + delete [] output_buffer; + + break; + } + + + case L_LTEXT16: // write out counted string. The count is in + // an octet, not a word. + + { + int output_count; // used as character count for binary output. + unsigned char lbcount; // To hold the leading byte count. + + // reserve enough for worst case + output_count = aValue.Length(); + // XXX I have commented the assert statement below out, because an LTEXT can't have length + // XXX bigger than 255, but sometimes this happens. A solution would be to throw an error + // XXX in structst.cpp, in the SimpleStructItem::SimpleStructItem() constructor. + UTF16 *output_buffer = new UTF16[output_count]; + + aValue.Export(output_buffer, output_count, SourceCharacterSet); + + // write out the count. + + lbcount = (unsigned char) output_count; + aStream.StreamIn(&lbcount,1); + + // write out the buffer as a stream of octets. + StreamOutCompressingIfReducesSize(aStream, output_buffer, output_count); + + delete [] output_buffer; + + break; + } + + + + + case L_BYTE: + case L_WORD: + case L_LONG: + case L_DOUBLE: + assert ( aValue.Length() > 0); + if(verbose) { MOFF; cout << "Converting " << aValue << " to number." << endl; MON;} + NumericValue( aValue, aItemType).StreamOut(aStream); + break; + case L_SRLINK: + NumericValue( aLinkValue, L_LONG).StreamOut(aStream); + break; + case L_LINK: + case L_LLINK: + { + if ( aValue.Length() > 0) + { + DataType dtype = ( aItemType == L_LINK) ? L_WORD : L_LONG; + if ( aValue.IsDecNatural() ) // If a resource id was specified explicitly e.g. -75 then output the value. + NumericValue(aValue, dtype).StreamOut(aStream); + else + { + // If FindId fails to find the link it will output an + // appropriate error and kill the program. + ErrorHandler::Register(aFileName, aLineNumber); + unsigned long id = pG->ResourceNameIds.FindId(aValue); + NumericValue(id, dtype).StreamOut(aStream); + } + } + break; + } + default: + { MOFF; cout << "RCBinaryStream:Unknown type " << gTypes.GetName( aItemType) << endl; MON;} + } + } + +void SimpleResourceItem::StreamOut(ResourceDataStream& aStream) const + { + StreamOutSingleValue(aStream, iValue, iLinkValue, iStructItem->iItemType, iFileName, iLineNumber); + } + +void ArrayResourceItem::StreamOut(ResourceDataStream& aStream) const + { + NumericValue * itemcount = NULL; + if(iStructItem->iSize.Length() == 0) // STRUCT item says [] i.e. no size specified. + { + if (iStructItem->iLenType == 0) // No LEN declaration so default to WORD. + itemcount = new NumericValue( L_WORD); + else + itemcount = new NumericValue( iStructItem->iLenType); // Use specified type. + + * itemcount = iValues.Size(); + itemcount->StreamOut(aStream); + + delete itemcount; + } + if ( iValues.Size() == 0) + return; + StringArrayIterator next( iValues); + String * p; + while ( ( p = next() ) != NULL) + { + assert( iStructItem->iItemType != L_SRLINK); // Can't handle arrays of SRlinks. + StreamOutSingleValue(aStream, *p, 0, iStructItem->iItemType, iFileName, iLineNumber); + } + } + +void StructTypeResourceItem::StreamOut(ResourceDataStream& aStream) const + { + iResourceItems.StreamOut(aStream); + } + +void StructArrayResourceItem::StreamOut(ResourceDataStream& aStream) const + { + NumericValue * itemcount = NULL; + if(iStructItem->iLenType == 0) // No LEN declaration so default to WORD. + itemcount = new NumericValue( L_WORD); + else + itemcount = new NumericValue( iStructItem->iLenType); // Use specified type. + * itemcount = iArrayOfResourceItemArrays.Size(); + itemcount->StreamOut(aStream); + delete itemcount; + if(iArrayOfResourceItemArrays.Size() == 0) + return; + iArrayOfResourceItemArrays.StreamOut(aStream); + } + +void ResourceItemArray::StreamOut(ResourceDataStream& aStream) const + { + if (Size() == 0) + return; + static unsigned int RecursionLevel = 0; // Top level should not have resource size output. + const bool outputSize=((iLenType != 0) && (RecursionLevel > 0)); + if (outputSize) + { + aStream.StartOfBlockWithSizePrefix(iLenType); + } + RecursionLevel++; + ResourceItemArrayIterator next(*this); + ResourceItem * p; + while ( ( p = next() ) != NULL) + { + p->StreamOut(aStream); + } + RecursionLevel--; + if (outputSize) + { + aStream.EndOfBlockWithSizePrefix(); + } + } + +void ResourceItemArrayArray::StreamOut(ResourceDataStream& aStream) const + { + if (Size() == 0) + return; + ResourceItemArrayArrayIterator next(*this); + ResourceItemArray * p; + while ( ( p = next() ) != NULL) + p->StreamOut(aStream); + }