os/persistentdata/persistentstorage/store/USTRM/US_UCMP.CPP
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
     1 // Copyright (c) 1998-2010 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Implementation of the Standard Compression Scheme for Unicode.
    15 // This code is compiled only in the Unicode build.
    16 // 
    17 //
    18 
    19 #ifdef _UNICODE
    20 
    21 #include <s32ucmp.h>
    22 
    23 const TUint32 TUnicodeCompressionState::iStaticWindow[EStaticWindows] =
    24 	{
    25 	0x0000,		// tags
    26 	0x0080,		// Latin-1 supplement
    27 	0x0100,		// Latin Extended-A
    28 	0x0300,		// Combining Diacritics
    29 	0x2000,		// General Punctuation
    30 	0x2080,		// Currency Symbols
    31 	0x2100,		// Letterlike Symbols and Number Forms
    32 	0x3000		// CJK Symbols and Punctuation
    33 	};
    34 
    35 const TUint32 TUnicodeCompressionState::iDynamicWindowDefault[EDynamicWindows] =
    36 	{
    37 	0x0080,		// Latin-1 supplement
    38 	0x00C0,		// parts of Latin-1 supplement and Latin Extended-A
    39 	0x0400,		// Cyrillic
    40 	0x0600,		// Arabic
    41 	0x0900,		// Devanagari
    42 	0x3040,		// Hiragana
    43 	0x30A0,		// Katakana
    44 	0xFF00		// Fullwidth ASCII
    45 	};
    46 
    47 const TUint16 TUnicodeCompressionState::iSpecialBase[ESpecialBases] =
    48 	{
    49 	0x00C0,		// Latin 1 letters (not symbols) and some of Extended-A
    50 	0x0250,		// IPA extensions
    51 	0x0370,		// Greek
    52 	0x0530,		// Armenian
    53 	0x3040,		// Hiragana
    54 	0x30A0,		// Katakana
    55 	0xFF60		// Halfwidth katakana
    56 	};
    57 
    58 // Single-byte mode tag values
    59 const TUint8 SQ0 = 0x01;	// <byte>				quote from window 0
    60 const TUint8 SDX = 0x0B;	// <hbyte> <lbyte>		define window in expansion area
    61 const TUint8 SQU = 0x0E;	// <hbyte> <lbyte>		quote Unicode value
    62 const TUint8 SCU = 0x0F;	//						switch to Unicode mode
    63 const TUint8 SC0 = 0x10;	//						select dynamic window 0
    64 const TUint8 SD0 = 0x18;	// <byte>				set dynamic window 0 index to <byte> and select it
    65 
    66 // Unicode mode tag values
    67 const TUint8 UC0 = 0xE0;	//						select dynamic window 0 and switch to single-byte mode
    68 const TUint8 UD0 = 0xE8;	// <byte>				set dynamic window 0 index to <byte>, select it and switch to
    69 							//						single-byte mode
    70 const TUint8 UQU = 0xF0;	// <hbyte>, <lbyte>		quote Unicode value
    71 const TUint8 UDX = 0xF1;	// <hbyte>, <lbyte>		define window in expansion area and switch to single-byte mode
    72 	
    73 TUnicodeCompressionState::TUnicodeCompressionState():
    74 	iUnicodeWords(0),
    75 	iMaxUnicodeWords(0),
    76 	iCompressedBytes(0),
    77 	iMaxCompressedBytes(0)
    78 	{
    79 	Reset();
    80 	}
    81 
    82 void TUnicodeCompressionState::Reset()
    83 	{
    84 	iUnicodeMode = FALSE;
    85 	iActiveWindowBase = 0x0080;
    86 	for (int i = 0; i < EDynamicWindows; i++)
    87 		iDynamicWindow[i] = iDynamicWindowDefault[i];
    88 	}
    89 
    90 
    91 // Return the index of the static window that contains this code, if any, or -1 if there is none.
    92 TInt TUnicodeCompressionState::StaticWindowIndex(TUint16 aCode)
    93 	{
    94 	for (TInt i = 0; i < EStaticWindows; i++)
    95 		if (aCode >= iStaticWindow[i] && aCode < iStaticWindow[i] + 128)
    96 			return i;
    97 	return -1;
    98 	}
    99 
   100 /*
   101 If aCode can be accommodated in one of the legal dynamic windows, return the index of that window
   102 in the offset table. If not return KErrNotFound.
   103 */
   104 TInt TUnicodeCompressionState::DynamicWindowOffsetIndex(TUint16 aCode)
   105 	{
   106 	if (aCode < 0x0080)
   107 		return KErrNotFound;
   108 	if (aCode >= 0x3400 && aCode <= 0xDFFF)
   109 		return KErrNotFound;
   110 
   111 	/*
   112 	Prefer sections that cross half-block boundaries. These are better adapted to actual text.
   113 	They are represented by offset indices 0xf9..0xff.
   114 	*/
   115 	for (int i = 0; i < ESpecialBases; i++)
   116 		if (aCode >= iSpecialBase[i] && aCode < iSpecialBase[i] + 128)
   117 			return 0xF9 + i;
   118 
   119 	/*
   120 	Offset indices 0x01..0x67 represent half blocks from 0x0080 to 0x3380 and
   121 	0x68..0xA7 represent half blocks from 0xE000 to 0xFF80.
   122 	*/
   123 	if (aCode >= 0xE000)
   124 		aCode -= 0xAC00;
   125 	return aCode / 0x80;
   126 	}
   127 
   128 // Return the base of the window represented by offset index <n>. Return 0 if the offset index is illegal.
   129 TUint32 TUnicodeCompressionState::DynamicWindowBase(TInt aOffsetIndex)
   130 	{
   131 	if (aOffsetIndex >= 0xF9 && aOffsetIndex <= 0xFF)
   132 		{
   133 		/*
   134 		WARNING: don't optimise the following two lines by replacing them with
   135 		'return iSpecialBase[aOffsetIndex - 0xF9];'. To do so would re-introduce an error
   136 		in ARM builds caused by optimisation and consequent erroneous fixing up
   137 		of the array base: see defect EDNGASR-4AGJQX in ER5U defects.
   138 		*/
   139 		int special_base_index = aOffsetIndex - 0xF9;
   140 		return iSpecialBase[special_base_index];
   141 		}
   142 	if (aOffsetIndex >= 0x01 && aOffsetIndex <= 0x67)
   143 		return aOffsetIndex * 0x80;
   144 	if (aOffsetIndex >= 0x68 && aOffsetIndex <= 0xA7)
   145 		return aOffsetIndex * 0x80 + 0xAC00;
   146 	return 0;
   147 	}
   148 
   149 TBool TUnicodeCompressionState::EncodeAsIs(TUint16 aCode)
   150 	{
   151 	return aCode == 0x0000 || aCode == 0x0009 || aCode == 0x000A || aCode == 0x000D ||
   152 		   (aCode >= 0x0020 && aCode <= 0x007F);
   153 	}
   154 
   155 #pragma BullseyeCoverage off
   156 
   157 void TUnicodeCompressionState::Panic(TPanic aPanic)
   158 	{
   159 	User::Panic(_L("ucmp"),aPanic);
   160 	}
   161 
   162 #pragma BullseyeCoverage on
   163 
   164 EXPORT_C TUnicodeCompressor::TUnicodeCompressor():
   165 	iInputBufferStart(0),
   166 	iInputBufferSize(0),
   167 	iOutputBufferStart(0),
   168 	iOutputBufferSize(0),
   169 	iDynamicWindowIndex(0),
   170 	iOutputStream(NULL),
   171 	iOutputPointer(NULL),
   172 	iInput(NULL)
   173 	{
   174 	}
   175 
   176 EXPORT_C void TUnicodeCompressor::CompressL(RWriteStream& aOutput,MUnicodeSource& aInput,
   177 											TInt aMaxOutputBytes,TInt aMaxInputWords,
   178 											TInt* aOutputBytes,TInt* aInputWords)
   179 	{
   180 	DoCompressL(&aOutput,NULL,&aInput,aMaxOutputBytes,aMaxInputWords,aOutputBytes,aInputWords);
   181 	}
   182 
   183 EXPORT_C void TUnicodeCompressor::CompressL(TUint8* aOutput,MUnicodeSource& aInput,
   184 											TInt aMaxOutputBytes,TInt aMaxInputWords,
   185 											TInt* aOutputBytes,TInt* aInputWords)
   186 	{
   187 	DoCompressL(NULL,aOutput,&aInput,aMaxOutputBytes,aMaxInputWords,aOutputBytes,aInputWords);
   188 	}
   189 
   190 EXPORT_C TInt TUnicodeCompressor::FlushL(RWriteStream& aOutput,TInt aMaxOutputBytes,TInt& aOutputBytes)
   191 	{
   192 	DoCompressL(&aOutput,NULL,NULL,aMaxOutputBytes,0,&aOutputBytes,NULL);
   193 	return iOutputBufferSize;
   194 	}
   195 
   196 EXPORT_C TInt TUnicodeCompressor::FlushL(TUint8* aOutput,TInt aMaxOutputBytes,TInt& aOutputBytes)
   197 	{
   198 	DoCompressL(NULL,aOutput,NULL,aMaxOutputBytes,0,&aOutputBytes,NULL);
   199 	return iOutputBufferSize;
   200 	}
   201 
   202 EXPORT_C TInt TUnicodeCompressor::CompressedSizeL(MUnicodeSource& aInput,TInt aInputWords)
   203 	{
   204 	TInt bytes;
   205 	TUnicodeCompressor c;
   206 	c.DoCompressL(NULL,NULL,&aInput,KMaxTInt,aInputWords,&bytes,NULL);
   207 	return bytes;
   208 	}
   209 
   210 // Compress until input or output is exhausted or an exception occurs.
   211 void TUnicodeCompressor::DoCompressL(RWriteStream* aOutputStream,TUint8* aOutputPointer,MUnicodeSource* aInput,
   212 									 TInt aMaxOutputBytes,TInt aMaxInputWords,
   213 									 TInt* aOutputBytes,TInt* aInputWords)
   214 	{
   215 	iOutputStream = aOutputStream;
   216 	iOutputPointer = aOutputPointer;
   217 	iInput = aInput;
   218 	iMaxCompressedBytes = aMaxOutputBytes;
   219 	iMaxUnicodeWords = aMaxInputWords;
   220 	iCompressedBytes = iUnicodeWords = 0;
   221 	FlushOutputBufferL();
   222 	if (iInput)
   223 		{
   224 		while (iUnicodeWords < iMaxUnicodeWords && iCompressedBytes < iMaxCompressedBytes)
   225 			{
   226 			TUint16 x = iInput->ReadUnicodeValueL();
   227 			TAction action(x);
   228 			iInputBuffer[(iInputBufferStart + iInputBufferSize) % EMaxInputBufferSize] = action;
   229 			iInputBufferSize++;
   230 			iUnicodeWords++;
   231 			if (iInputBufferSize == EMaxInputBufferSize)
   232 				WriteRunL();
   233 			}
   234 		}
   235 	FlushInputBufferL();
   236 	if (aOutputBytes)
   237 		*aOutputBytes = iCompressedBytes;
   238 	if (aInputWords)
   239 		*aInputWords = iUnicodeWords;
   240 	}
   241 
   242 TUnicodeCompressor::TAction::TAction(TUint16 aCode):
   243 	iCode(aCode)
   244 	{
   245 	if (TUnicodeCompressionState::EncodeAsIs(aCode))
   246 		iTreatment = EPlainASCII;
   247 	else
   248 		{
   249 		iTreatment = TUnicodeCompressionState::DynamicWindowOffsetIndex(aCode);
   250 		if (iTreatment == -1)
   251 			{
   252 			iTreatment = TUnicodeCompressionState::StaticWindowIndex(aCode);
   253 			if (iTreatment == -1)
   254 				iTreatment = EPlainUnicode;
   255 			else
   256 				iTreatment += EFirstStatic;
   257 			}
   258 		}
   259 	}
   260 
   261 void TUnicodeCompressor::WriteCharacterFromBuffer()
   262 	{
   263 	const TAction& action = iInputBuffer[iInputBufferStart];
   264 	iInputBufferSize--;
   265 	iInputBufferStart = (iInputBufferStart + 1) % EMaxInputBufferSize;
   266 	WriteCharacter(action);
   267 	}
   268 
   269 void TUnicodeCompressor::FlushInputBufferL()
   270 	{
   271 	while (iInputBufferSize > 0 && iCompressedBytes < iMaxCompressedBytes)
   272 		WriteRunL();
   273 	}
   274 
   275 void TUnicodeCompressor::WriteRunL()
   276 	{
   277 	// Write out any leading characters that can be passed through.
   278 	if (!iUnicodeMode)
   279 		while (iInputBufferSize > 0)
   280 			{
   281 			const TAction& action = iInputBuffer[iInputBufferStart];
   282 			if (action.iTreatment == TAction::EPlainASCII ||
   283 				(action.iCode >= iActiveWindowBase && action.iCode < iActiveWindowBase + 128))
   284 				WriteCharacterFromBuffer();
   285 			else
   286 				break;
   287 			}
   288 
   289 	// Write a run of characters that cannot be passed through.
   290 	int i;
   291 	if (iInputBufferSize > 0)
   292 		{
   293 		/*
   294 		Find a run of characters with the same treatment and select that treatment
   295 		if the run has more than one character.
   296 		*/
   297 		int treatment = iInputBuffer[iInputBufferStart].iTreatment;
   298 		int next_treatment = treatment;
   299 		int run_size = 1;
   300 		for (i = 1; i < iInputBufferSize; i++)
   301 			{
   302 			int index = (iInputBufferStart + i) % EMaxInputBufferSize;
   303 			next_treatment = iInputBuffer[index].iTreatment;
   304 			if (next_treatment != treatment)
   305 				break;
   306 			run_size++;
   307 			}
   308 		if (run_size > 1)
   309 			SelectTreatment(treatment);
   310 		for (i = 0; i < run_size; i++)
   311 			WriteCharacterFromBuffer();
   312 		}
   313 
   314 	FlushOutputBufferL();
   315 	}
   316 
   317 void TUnicodeCompressor::FlushOutputBufferL()
   318 	{
   319 	while (iOutputBufferSize > 0 &&	iCompressedBytes < iMaxCompressedBytes)
   320 		{
   321 		TUint8 byte = iOutputBuffer[iOutputBufferStart];
   322 		if (iOutputPointer)
   323 			*iOutputPointer++ = byte;
   324 		else if (iOutputStream)
   325 			iOutputStream->WriteUint8L(byte);
   326 		iCompressedBytes++;
   327 		iOutputBufferSize--;
   328 		iOutputBufferStart = (iOutputBufferStart + 1) % EMaxOutputBufferSize;
   329 		}
   330 	}
   331 
   332 void TUnicodeCompressor::SelectTreatment(TInt aTreatment)
   333 	{
   334 	if (aTreatment == TAction::EPlainUnicode)
   335 		{
   336 		// Switch to Unicode mode if not there already.
   337 		if (!iUnicodeMode)
   338 			{
   339 			WriteByte(SCU);
   340 			iUnicodeMode = TRUE;
   341 			}
   342 		return;
   343 		}
   344 
   345 	if (aTreatment == TAction::EPlainASCII)
   346 		{
   347 		// Switch to single-byte mode, using the current dynamic window, if not there already.
   348 		if (iUnicodeMode)
   349 			{
   350 			WriteByte(UC0 + iDynamicWindowIndex);
   351 			iUnicodeMode = FALSE;
   352 			}
   353 		return;
   354 		}
   355 
   356 	if (aTreatment >= TAction::EFirstDynamic && aTreatment <= TAction::ELastDynamic)
   357 		{
   358 		TUint32 base = DynamicWindowBase(aTreatment);
   359 
   360 		// Switch to the appropriate dynamic window if it is available; if not, redefine and select dynamic window 4.
   361 		for (int i = 0; i < EDynamicWindows; i++)
   362 			if (base == iDynamicWindow[i])
   363 				{
   364 				if (iUnicodeMode)
   365 					WriteByte(UC0 + i);
   366 				else if (i != iDynamicWindowIndex)
   367 					WriteByte(SC0 + i);
   368 				iUnicodeMode = FALSE;
   369 				iDynamicWindowIndex = i;
   370 				iActiveWindowBase = base;
   371 				return;
   372 				}
   373 		if (iUnicodeMode)
   374 			WriteByte(UD0 + 4);
   375 		else
   376 			WriteByte(SD0 + 4);
   377 		iDynamicWindowIndex = 4;
   378 		iUnicodeMode = FALSE;
   379 		WriteByte(aTreatment);
   380 		iDynamicWindow[4] = base;
   381 		iActiveWindowBase = base;
   382 		return;
   383 		}
   384 	}
   385 
   386 // Write a character without changing mode or window.
   387 void TUnicodeCompressor::WriteCharacter(const TAction& aAction)
   388 	{
   389 	if (iUnicodeMode)
   390 		WriteUCharacter(aAction.iCode);
   391 	else
   392 		WriteSCharacter(aAction);
   393 	}
   394 
   395 void TUnicodeCompressor::WriteUCharacter(TUint16 aCode)
   396 	{
   397 	// Emit the 'quote Unicode' tag if the character would conflict with a tag.
   398 	if (aCode >= 0xE000 && aCode <= 0xF2FF)
   399 		WriteByte(UQU);
   400 
   401 	// Write the Unicode value big-end first.
   402 	WriteByte((aCode >> 8) & 0xFF);
   403 	WriteByte(aCode & 0xFF);
   404 	}
   405 
   406 void TUnicodeCompressor::WriteByte(TUint aByte)
   407 	{
   408 	if (iOutputBufferSize >= EMaxOutputBufferSize)
   409 		Panic(EOutputBufferOverflow); //Panic here is ok as this is a programming error
   410 	iOutputBuffer[(iOutputBufferStart + iOutputBufferSize) % EMaxOutputBufferSize] = (TUint8)aByte;
   411 	iOutputBufferSize++;
   412 	}
   413 
   414 void TUnicodeCompressor::WriteSCharacter(const TAction& aAction)
   415 	{
   416 	// Characters in the range 0x0020..0x007F, plus nul, tab, cr, and lf, can be emitted as their low bytes.
   417 	if (aAction.iTreatment == TAction::EPlainASCII)
   418 		{
   419 		WriteByte(aAction.iCode);
   420 		return;
   421 		}
   422 
   423 	// Characters in a static window can be written using SQ<n> plus a byte in the range 0x00-0x7F
   424 	if (aAction.iTreatment >= TAction::EFirstStatic && aAction.iTreatment <= TAction::ELastStatic)
   425 		{
   426 		int window = aAction.iTreatment - TAction::EFirstStatic;
   427 		WriteByte(SQ0 + window);
   428 		WriteByte(aAction.iCode);
   429 		return;
   430 		}
   431 
   432 	// Characters in the current dynamic window can be written as a byte in the range 0x80-0xFF.
   433 	if (aAction.iCode >= iActiveWindowBase && aAction.iCode < iActiveWindowBase + 128)
   434 		{
   435 		WriteByte(aAction.iCode - iActiveWindowBase + 0x80);
   436 		return;
   437 		}
   438 
   439 	// Characters in another dynamic window can be written using SQ<n> plus a byte in the range 0x80-0xFF
   440 	int i;
   441 	for (i = 0; i < EDynamicWindows; i++)
   442 		if (aAction.iCode >= iDynamicWindow[i] && aAction.iCode < iDynamicWindow[i] + 128)
   443 			{
   444 			WriteByte(SQ0 + i);
   445 			WriteByte(aAction.iCode - iDynamicWindow[i] + 0x80);
   446 			return;
   447 			}
   448 
   449 	// Other characters can be quoted.
   450 	WriteByte(SQU);
   451 	WriteByte((aAction.iCode >> 8) & 0xFF);
   452 	WriteByte(aAction.iCode & 0xFF);
   453 	return;
   454 	}
   455 
   456 EXPORT_C TUnicodeExpander::TUnicodeExpander():
   457 	iInputBufferStart(0),
   458 	iInputBufferSize(0),
   459 	iOutputBufferStart(0),
   460 	iOutputBufferSize(0),
   461 	iOutput(NULL),
   462 	iInputStream(NULL),
   463 	iInputPointer(NULL)
   464 	{
   465 	}
   466 
   467 EXPORT_C void TUnicodeExpander::ExpandL(MUnicodeSink& aOutput,RReadStream& aInput,
   468 										TInt aMaxOutputWords,TInt aMaxInputBytes,
   469 										TInt* aOutputWords,TInt* aInputBytes)
   470 	{
   471 	DoExpandL(&aOutput,&aInput,NULL,aMaxOutputWords,aMaxInputBytes,aOutputWords,aInputBytes);
   472 	}
   473 
   474 EXPORT_C void TUnicodeExpander::ExpandL(MUnicodeSink& aOutput,const TUint8* aInput,
   475 										TInt aMaxOutputWords,TInt aMaxInputBytes,
   476 										TInt* aOutputWords,TInt* aInputBytes)
   477 	{
   478 	DoExpandL(&aOutput,NULL,aInput,aMaxOutputWords,aMaxInputBytes,aOutputWords,aInputBytes);
   479 	}
   480 
   481 EXPORT_C TInt TUnicodeExpander::FlushL(MUnicodeSink& aOutput,TInt aMaxOutputWords,TInt& aOutputWords)
   482 	{
   483 	DoExpandL(&aOutput,NULL,NULL,aMaxOutputWords,0,&aOutputWords,NULL);
   484 	return iOutputBufferSize;
   485 	}
   486 
   487 EXPORT_C TInt TUnicodeExpander::ExpandedSizeL(RReadStream& aInput,TInt aInputBytes)
   488 	{
   489 	TInt words;
   490 	TUnicodeExpander e;
   491 	e.DoExpandL(NULL,&aInput,NULL,KMaxTInt,aInputBytes,&words,NULL);
   492 	return words;
   493 	}
   494 
   495 EXPORT_C TInt TUnicodeExpander::ExpandedSizeL(const TUint8* aInput,TInt aInputBytes)
   496 	{
   497 	TInt words;
   498 	TUnicodeExpander e;
   499 	e.DoExpandL(NULL,NULL,aInput,KMaxTInt,aInputBytes,&words,NULL);
   500 	return words;
   501 	}
   502 
   503 // Expand until input or output is exhausted or an exception occurs.
   504 void TUnicodeExpander::DoExpandL(MUnicodeSink* aOutput,RReadStream* aInputStream,const TUint8* aInputPointer,
   505 								 TInt aMaxOutputWords,TInt aMaxInputBytes,
   506 								 TInt* aOutputWords,TInt* aInputBytes)
   507 	{
   508 	iOutput = aOutput;
   509 	iInputStream = aInputStream;
   510 	iInputPointer = aInputPointer;
   511 	iMaxUnicodeWords = aMaxOutputWords;
   512 	iMaxCompressedBytes = aMaxInputBytes;
   513 	iUnicodeWords = iCompressedBytes = 0;
   514 	iInputBufferStart = 0;
   515 	FlushOutputBufferL();
   516 	if (iInputPointer || iInputStream)
   517 		{
   518 		while (iUnicodeWords + iOutputBufferSize < iMaxUnicodeWords && iCompressedBytes < iMaxCompressedBytes)
   519 			HandleByteL();
   520 		}
   521 	if (aOutputWords)
   522 		*aOutputWords = iUnicodeWords;
   523 	if (aInputBytes)
   524 		*aInputBytes = iCompressedBytes;
   525 	}
   526 
   527 void TUnicodeExpander::HandleByteL()
   528 	{
   529 	TUint8 byte;
   530 	TBool handled = FALSE;
   531 	if (ReadByteL(byte))
   532 		{
   533 		if (iUnicodeMode)
   534 			handled = HandleUByteL(byte);
   535 		else
   536 			handled = HandleSByteL(byte);
   537 		}
   538 	iInputBufferStart = 0;
   539 	if (handled)
   540 		iInputBufferSize = 0;
   541 	FlushOutputBufferL();
   542 	}
   543 
   544 void TUnicodeExpander::FlushOutputBufferL()
   545 	{
   546 	while (iOutputBufferSize > 0 &&	iUnicodeWords < iMaxUnicodeWords)
   547 		{
   548 		if (iOutput)
   549 			iOutput->WriteUnicodeValueL(iOutputBuffer[iOutputBufferStart]);
   550 		iUnicodeWords++;
   551 		iOutputBufferSize--;
   552 		iOutputBufferStart = (iOutputBufferStart + 1) % EMaxOutputBufferSize;
   553 		}
   554 	}
   555 
   556 TBool TUnicodeExpander::HandleSByteL(TUint8 aByte)
   557 	{
   558 	// 'Pass-through' codes.
   559 	if (TUnicodeCompressionState::EncodeAsIs(aByte))
   560 		{
   561 		WriteChar(aByte);
   562 		return TRUE;
   563 		}
   564 
   565 	// Codes 0x80-0xFF select a character from the active window.
   566 	if (aByte >= 0x80)
   567 		{
   568 		WriteChar32(iActiveWindowBase + aByte - 0x80);
   569 		return TRUE;
   570 		}
   571 
   572 	// SQU: quote a Unicode character.
   573 	if (aByte == SQU)
   574 		return QuoteUnicodeL();
   575 
   576 	// SCU: switch to Unicode mode.
   577 	if (aByte == SCU)
   578 		{
   579 		iUnicodeMode = TRUE;
   580 		return TRUE;
   581 		}
   582 
   583 	// SQn: quote from window n.
   584 	if (aByte >= SQ0 && aByte <= SQ0 + 7)
   585 		{
   586 		int window = aByte - SQ0;
   587 		TUint8 byte;
   588 		if (ReadByteL(byte))
   589 			{
   590 			TUint32 c = byte;
   591 			if (c <= 0x7F)
   592 				c += iStaticWindow[window];
   593 			else
   594 				c += iDynamicWindow[window] - 0x80;
   595 			WriteChar32(c);
   596 			return TRUE;
   597 			}
   598 		else
   599 			return FALSE;
   600 		}
   601 
   602 	// SCn: switch to dynamic window n.
   603 	if (aByte >= SC0 && aByte <= SC0 + 7)
   604 		{
   605 		iActiveWindowBase = iDynamicWindow[aByte - SC0];
   606 		return TRUE;
   607 		}
   608 
   609 	// SDn: define dynamic window n and switch to it.
   610 	if (aByte >= SD0 && aByte <= SD0 + 7)
   611 		return DefineWindowL(aByte - SD0);
   612 
   613 	// SDX: define window in the expansion space.
   614 	if (aByte == SDX)
   615 		return DefineExpansionWindowL();
   616 
   617 	User::Leave(KErrCorrupt);
   618 	return FALSE;
   619 	}
   620 
   621 TBool TUnicodeExpander::HandleUByteL(TUint8 aByte)
   622 	{
   623 	// Plain Unicode; get the low byte and emit the Unicode value.
   624 	if (aByte <= 0xDF || aByte >= 0xF3)
   625 		{
   626 		TUint8 lo;
   627 		if (ReadByteL(lo))
   628 			{
   629 			TUint16 c = (TUint16)((aByte << 8) | lo);
   630 			WriteChar(c);
   631 			return TRUE;
   632 			}
   633 		else
   634 			return FALSE;
   635 		}
   636 
   637 	// Quote a Unicode character that would otherwise conflict with a tag.
   638 	if (aByte == UQU)
   639 		return QuoteUnicodeL();
   640 
   641 	// UCn: change to single byte mode and select window n.
   642 	if (aByte >= UC0 && aByte <= UC0 + 7)
   643 		{
   644 		iUnicodeMode = FALSE;
   645 		iActiveWindowBase = iDynamicWindow[aByte - UC0];
   646 		return TRUE;
   647 		}
   648 
   649 	// UDn: define dynamic window n and switch to it.
   650 	if (aByte >= UD0 && aByte <= UD0 + 7)
   651 		return DefineWindowL(aByte - UD0);
   652 
   653 	// UDX: define window in the expansion space.
   654 	if (aByte == UDX)
   655 		return DefineExpansionWindowL();
   656 
   657 	User::Leave(KErrCorrupt);
   658 	return FALSE;
   659 	}
   660 
   661 TBool TUnicodeExpander::QuoteUnicodeL()
   662 	{
   663 	TUint8 hi, lo;
   664 	if (ReadByteL(hi) && ReadByteL(lo))
   665 		{
   666 		TUint16 c = (TUint16)((hi << 8) | lo);
   667 		WriteChar(c);
   668 		return TRUE;
   669 		}
   670 	else
   671 		return FALSE;
   672 	}
   673 
   674 TBool TUnicodeExpander::DefineWindowL(TInt aIndex)
   675 	{
   676 	TUint8 window;
   677 	if (ReadByteL(window))
   678 		{
   679 		iUnicodeMode = FALSE;
   680 		iActiveWindowBase = DynamicWindowBase(window);
   681 		iDynamicWindow[aIndex] = iActiveWindowBase;
   682 		return TRUE;
   683 		}
   684 	else
   685 		return FALSE;
   686 	}
   687 
   688 TBool TUnicodeExpander::DefineExpansionWindowL()
   689 	{
   690 	TUint8 hi, lo;
   691 	if (ReadByteL(hi) && ReadByteL(lo))
   692 		{
   693 		iUnicodeMode = FALSE;
   694 		iActiveWindowBase = 0x10000 + (0x80 * ((hi & 0x1F) * 0x100 + lo));
   695 		iDynamicWindow[hi >> 5] = iActiveWindowBase;
   696 		return TRUE;
   697 		}
   698 	else
   699 		return FALSE;
   700 	}
   701 
   702 // Read either from the buffer (in the case of restarting after source finished in mid-operation) or from the source.
   703 TBool TUnicodeExpander::ReadByteL(TUint8& aByte)
   704 	{
   705 	if (iInputBufferStart < iInputBufferSize)
   706 		{
   707 		aByte = iInputBuffer[iInputBufferStart++];
   708 		return TRUE;
   709 		}
   710 	else if (iCompressedBytes < iMaxCompressedBytes)
   711 		{
   712 		if (iInputPointer)
   713 			aByte = *iInputPointer++;
   714 		else
   715 			aByte = iInputStream->ReadUint8L();
   716 		iInputBuffer[iInputBufferStart++] = aByte;
   717 		iInputBufferSize = iInputBufferStart;
   718 		iCompressedBytes++;
   719 		return TRUE;
   720 		}
   721 	else
   722 		return FALSE;
   723 	}
   724 
   725 void TUnicodeExpander::WriteChar(TUint16 aChar)
   726 	{
   727 	if (iOutputBufferSize >= EMaxOutputBufferSize)
   728 		Panic(EOutputBufferOverflow); //Panic here is ok since this is a programming error
   729 	iOutputBuffer[(iOutputBufferStart + iOutputBufferSize) % EMaxOutputBufferSize] = aChar;
   730 	iOutputBufferSize++;
   731 	}
   732 
   733 // Write a Unicode character; write using surrogates if in the range 0x10000..0x10FFFF.
   734 void TUnicodeExpander::WriteChar32(TUint aChar)
   735 	{
   736 	if (aChar <= 0xFFFF)
   737 		WriteChar((TUint16)aChar);
   738 	else if (aChar <= 0x10FFFF)
   739 		{
   740 		aChar -= 0x10000;									// reduce to 20-bit value in the range 0x0..0xFFFFF
   741 		WriteChar((TUint16)(0xD800 + (aChar >> 10)));		// first high surrogate + high 10 bits
   742 		WriteChar((TUint16)(0xDC00 + (aChar & 0x03FF)));	// first low surrogate + low 10 bits
   743 		}
   744 	else
   745 		//Panic to be kept here as impossible to test this case (nor the one before). Biggest value that can be passed is 0xFFFFF 
   746 		Panic(ENotUnicode);
   747 	}
   748 
   749 #endif // _UNICODE