bintools/rcomp/src/RCOSTRM.CPP
changeset 2 39c28ec933dd
--- /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 <assert.h>
+#include <stdlib.h>
+#include <memory.h>
+
+#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) ? "<none>" : 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) ? "<none>" : gTypes.GetName( o.iLenType) );
+	os << "\tSize: " << o.iSize;
+	os << endl;
+	return os;
+	}
+
+ostream& operator<< ( ostream & os, StructItemArray & o)
+	{
+	if (o.Size() == 0)
+		return ( os << "<none>");
+	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) ? "<none>" : gTypes.GetName( o.iLenType) ) << endl;
+	os << o.iSIA;
+	return os;
+	}
+
+ostream& operator<< ( ostream & os, StructHeaderArray & o)
+	{
+	if (o.Size() == 0)
+		return ( os << "<none>");
+	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 << "<none>");
+	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 << "<none>");
+	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);
+	}