os/textandloc/charconvfw/charconvplugins/src/plugins/iscii.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 /*
     2 * Copyright (c) 2002 Nokia Corporation and/or its subsidiary(-ies).
     3 * All rights reserved.
     4 * This component and the accompanying materials are made available
     5 * under the terms of "Eclipse Public License v1.0"
     6 * which accompanies this distribution, and is available
     7 * at the URL "http://www.eclipse.org/legal/epl-v10.html".
     8 *
     9 * Initial Contributors:
    10 * Nokia Corporation - initial contribution.
    11 *
    12 * Contributors:
    13 *
    14 * Description:     Implements the characterconversion plug-in
    15 *                for ISCII characterset.
    16 *
    17 */
    18 
    19 
    20 
    21 
    22 
    23 #include <e32std.h>
    24 #include <charconv.h>
    25 #include <convgeneratedcpp.h>
    26 #include <convutils.h>
    27 
    28 //The maximum length of any intermediate buffer allocated for conversion.
    29 const TInt KMaximumLengthOfIntermediateBuffer=5;
    30 //The ISCII ATR code point, used for ISCII script switching mechanism.
    31 const TUint KControlCharacterEscape=0xef;
    32 //The number of Indic scripts supported by the plug-in.
    33 //ISCII in general addresses all the Indic character sets.
    34 const TUint KNumberOfIndicCharactersetsSupported = 1;
    35 //The common reason for panic for all panics raised by the iscii plug-in
    36 _LIT16(KPanicReason,"ISCII Plug-in Panic"); 
    37 //The escape sequence for ISCII (ATR) is 0xEF and immidiate byte following 
    38 //that is the script selection code for Devanagari.
    39 _LIT8(KEscapeSequenceDevanagari,"\xef\x42");
    40 //The sequence for Explicit Halant, in unicode it gets converted to VIRAMA+ZWNJ
    41 _LIT8(KExplicitHalant,"\xe8\xe8");
    42 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    43 //sequence is converted to intermediate unused iscii code point.
    44 _LIT8(KReplacementForExplicitHalant,"\xe8\xfc");
    45 //The sequence for Soft Halant, in unicode it gets converted to VIRAMA+ZWJ
    46 _LIT8(KSoftHalant,"\xe8\xe9");
    47 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    48 //sequence is converted to intermediate unused iscii code point.
    49 _LIT8(KReplacementForSoftHalant,"\xe8\xfd");
    50 //Devanagari character Om
    51 _LIT8(KOm,"\xa1\xe9");
    52 ////For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    53 //sequence is converted to intermediate unused iscii code point.
    54 _LIT8(KReplacementForOm,"\xfe");
    55 //Devanagari character Avagraha
    56 _LIT8(KAvagraha,"\xea\xe9");
    57 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    58 //sequence is converted to intermediate unused iscii code point.
    59 _LIT8(KReplacementForAvagraha,"\xff");
    60 
    61 //Devanagari character VOCALIC RR
    62 _LIT8(KVocalicRr,"\xaa\xe9");
    63 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    64 //sequence is converted to intermediate unused iscii code point.
    65 _LIT8(KReplacementForVocalicRr,"\x80");
    66 //Devanagari character VOCALIC LL
    67 _LIT8(KVocalicLl,"\xa7\xe9");
    68 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    69 //sequence is converted to intermediate unused iscii code point.
    70 _LIT8(KReplacementForVocalicLl,"\x81");
    71 //Devanagari character VOCALIC L SIGN
    72 _LIT8(KVocalicLSign,"\xdb\xe9");
    73 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    74 //sequence is converted to intermediate unused iscii code point.
    75 _LIT8(KReplacementForVocalicLSign,"\x82");
    76 //Devanagari character VOCALIC LL SIGN
    77 _LIT8(KVocalicLlSign,"\xdc\xe9");
    78 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    79 //sequence is converted to intermediate unused iscii code point.
    80 _LIT8(KReplacementForVocalicLlSign,"\x83");
    81 //Devanagari character VOCALIC L
    82 _LIT8(KVocalicL,"\xa6\xe9");
    83 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    84 //sequence is converted to intermediate unused iscii code point.
    85 _LIT8(KReplacementForVocalicL,"\x84");
    86 //Devanagari character VOCALIC RR SIGN
    87 _LIT8(KVocalicRrSign,"\xdf\xe9");
    88 //For supportiing ISCII to Unicode conversion of multi byte ISCII sequences, the
    89 //sequence is converted to intermediate unused iscii code point.
    90 _LIT8(KReplacementForVocalicRrSign,"\x85");
    91 
    92 //Unconvertible ISCII character
    93 _LIT8(KIsciiUnconvertibleCharacter,"\xeb");
    94 
    95 enum TPanic
    96 {
    97 	//The panic raised by ConvertToUnicodeFromModalForeign_Internal() if the input
    98 	//conversion flag is CCnvCharacterSetConverter::EInputConversionFlagStopAtFirstUnconvertibleCharacter
    99 	EPanicBadInputConversionFlags=1,
   100 	//Panic raised if the buffer does not start with the escape sequence 0xEF
   101 	EPanicBadRemainderOfForeign,
   102 	//Panic is raised if the length of the search buffer is greater than the length of the 
   103 	//replacement buffer
   104 	EPanicBadReplacementBuffer,
   105 	//If the offset of start of the escape sequence is not an unsigned number.
   106 	EPanicBadStartOfNextEscapeSequence
   107 };
   108 
   109 //The dummy datastructure for the dummy conversion data i.e. used for conversion if the 
   110 //script selection code is not supported.
   111 #define ARRAY_LENGTH(aArray) (sizeof(aArray)/sizeof((aArray)[0]))
   112 
   113 LOCAL_D const SCnvConversionData::SOneDirectionData::SRange::UData::SKeyedTable1616::SEntry keyedTable1616_foreignToUnicode_1[]=
   114 	{
   115 		{
   116 		0xa0,
   117 		0xfffd
   118 		}
   119 	};
   120 
   121 LOCAL_D const SCnvConversionData::SOneDirectionData::SRange::UData::SKeyedTable1616::SEntry keyedTable1616_unicodeToForeign_1[]=
   122 	{
   123 		{
   124 		0xfffd,
   125 		0xa0
   126 		}
   127 	};
   128 
   129 LOCAL_D const SCnvConversionData::SVariableByteData::SRange foreignVariableByteDataRanges[]=
   130 	{
   131 		{
   132 		0x00,
   133 		0xff,
   134 		0,
   135 		0
   136 		}
   137 	};
   138 
   139 LOCAL_D const SCnvConversionData::SOneDirectionData::SRange foreignToUnicodeDataRanges[]=
   140 	{
   141 		{
   142 		0x00,
   143 		0x7f,
   144 		SCnvConversionData::SOneDirectionData::SRange::EDirect,
   145 		0,
   146 		0,
   147 			{
   148 			0
   149 			}
   150 		},
   151 		{
   152 		0xa0,
   153 		0xff,
   154 		SCnvConversionData::SOneDirectionData::SRange::EKeyedTable1616,
   155 		0,
   156 		0,
   157 			{
   158 			UData_SKeyedTable1616(keyedTable1616_foreignToUnicode_1)
   159 			}
   160 		}
   161 	};
   162 
   163 LOCAL_D const SCnvConversionData::SOneDirectionData::SRange unicodeToForeignDataRanges[]=
   164 	{
   165 		{
   166 		0x0000,
   167 		0x007f,
   168 		SCnvConversionData::SOneDirectionData::SRange::EDirect,
   169 		1,
   170 		0,
   171 			{
   172 			0
   173 			}
   174 		},
   175 		{
   176 		0x00a0,
   177 		0xffff,
   178 		SCnvConversionData::SOneDirectionData::SRange::EKeyedTable1616,
   179 		1,
   180 		0,
   181 			{
   182 			UData_SKeyedTable1616(keyedTable1616_unicodeToForeign_1)
   183 			}
   184 		}
   185 	};
   186 
   187 //The dummy conversion data to be used for conversion if the iscii code sequence is not
   188 //Devanagari (i.e. the script selection code is not 0x42 and something else.
   189 //In this case the ISCII characters are converted to unconvertible characters.
   190 
   191 LOCAL_D const SCnvConversionData conversionDataDummy=
   192 	{
   193 	SCnvConversionData::EFixedBigEndian,
   194 		{
   195 		ARRAY_LENGTH(foreignVariableByteDataRanges),
   196 		foreignVariableByteDataRanges
   197 		},
   198 		{
   199 		ARRAY_LENGTH(foreignToUnicodeDataRanges),
   200 		foreignToUnicodeDataRanges
   201 		},
   202 		{
   203 		ARRAY_LENGTH(unicodeToForeignDataRanges),
   204 		unicodeToForeignDataRanges
   205 		},
   206 	NULL,
   207 	NULL
   208 	};
   209 
   210 
   211 
   212 #ifdef EKA2
   213 
   214 ///////////////////////////////////////////////////////////////
   215 // 3.1 Code
   216 
   217 // INCLUDES
   218 #include <ecom/implementationproxy.h>
   219 #include <charactersetconverter.h>
   220 
   221 
   222 /**
   223 * The character conversion plug-in implementation for Iscii.
   224 *
   225 *  @lib ecom.lib
   226 *  @since Series 60 3.1
   227 */
   228 
   229 class CIsciiImplementation : public CCharacterSetConverterPluginInterface
   230 {
   231     public:
   232     	//From CCharacterSetConverterPluginInterface
   233         virtual const TDesC8& ReplacementForUnconvertibleUnicodeCharacters();
   234 
   235 	//From CCharacterSetConverterPluginInterface
   236         virtual TInt ConvertFromUnicode(
   237             CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters, 
   238             const TDesC8& aReplacementForUnconvertibleUnicodeCharacters, 
   239             TDes8& aForeign, 
   240             const TDesC16& aUnicode, 
   241             CCnvCharacterSetConverter::TArrayOfAscendingIndices& aIndicesOfUnconvertibleCharacters );
   242 
   243 	//From CCharacterSetConverterPluginInterface
   244         virtual TInt ConvertToUnicode(
   245             CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters, 
   246             TDes16& aUnicode, 
   247             const TDesC8& aForeign, 
   248             TInt&, 
   249             TInt& aNumberOfUnconvertibleCharacters, 
   250             TInt& aIndexOfFirstByteOfFirstUnconvertibleCharacter );
   251 
   252 	//From CCharacterSetConverterPluginInterface
   253         virtual TBool IsInThisCharacterSetL(
   254             TBool& aSetToTrue, 
   255             TInt& aConfidenceLevel, 
   256             const TDesC8& );
   257 
   258         static CIsciiImplementation* NewL();
   259 
   260         virtual ~CIsciiImplementation();
   261     private:
   262         CIsciiImplementation();
   263 };
   264 
   265 //Checks if a descriptor starts with another descriptor at the begining.
   266 LOCAL_C TBool IsStartOf(const TDesC8& aEscapeSequence, const TDesC8& aBuffer)
   267 {
   268 	const TInt lengthOfStart=aEscapeSequence.Length();
   269 	return (aBuffer.Length()>=lengthOfStart) && (aBuffer.Left(lengthOfStart)==aEscapeSequence);
   270 }
   271 // -----------------------------------------------------------------------------
   272 // MatchesEscapeSequence()
   273 //If the remainder of the foreign text starts with the passed escapesequence, modifies the remainder of the foreign text
   274 //and  sets the homogeneous run buffer that uses the same conversion data.
   275 //The homogeneous run buffer is the buffer that will use a single conversion data, it doesn't contain the attribute code
   276 // neither it contains the script switching code.
   277 //The aRemainderOfForeign buffer 
   278 
   279 // Returns: ETrue: If the sequence contains the escape sequence
   280 //              EFalse: If the sequence does not contain the escape sequence
   281 //
   282 // -----------------------------------------------------------------------------
   283 //
   284 LOCAL_C TBool MatchesEscapeSequence(TInt& aNumberOfForeignBytesConsumed, TPtrC8& aHomogeneousRun,
   285 		TPtrC8& aRemainderOfForeign, const TDesC8& aEscapeSequence)
   286 {
   287 	const TInt lengthOfEscapeSequence=aEscapeSequence.Length();
   288 	if (IsStartOf(aEscapeSequence, aRemainderOfForeign))
   289 		{
   290 		aRemainderOfForeign.Set(aRemainderOfForeign.Mid(lengthOfEscapeSequence));
   291 		const TInt startOfNextEscapeSequence=aRemainderOfForeign.Locate(KControlCharacterEscape);
   292 		if (startOfNextEscapeSequence==KErrNotFound)
   293 			{
   294 			aHomogeneousRun.Set(aRemainderOfForeign);
   295 			aRemainderOfForeign.Set(NULL, 0);
   296 			}
   297 		else
   298 			{
   299 			aHomogeneousRun.Set(aRemainderOfForeign.Left(startOfNextEscapeSequence));
   300 			aRemainderOfForeign.Set(aRemainderOfForeign.Mid(startOfNextEscapeSequence));
   301 			}
   302 		aNumberOfForeignBytesConsumed+=lengthOfEscapeSequence;
   303 		return ETrue;
   304 		}
   305 	return EFalse;
   306 }
   307 
   308 // -----------------------------------------------------------------------------
   309 // NextHomogeneousForeignRun()
   310 //Matches the escape sequence of each of the elements of the SState array with the remainder of
   311 //foreign text and if the escape sequence matches with the start of remainder of the foreign text, 
   312 //then the conversion data is set to the conversion data corresponding to the escape sequence
   313 //Also the homogeneous foreign text for conversion with the same escape sequence is set.
   314 
   315 // Returns: ETrue: If length of the remainder of foreign buffer is nonzero.
   316 //              EFalse: If length of the remainder of foreign buffer is zero.
   317 //
   318 // -----------------------------------------------------------------------------
   319 //
   320 
   321 
   322 LOCAL_C TBool NextHomogeneousForeignRun(const SCnvConversionData*& aConversionData, TInt& aNumberOfForeignBytesConsumed, TPtrC8& aHomogeneousRun, TPtrC8& aRemainderOfForeign, const TArray<CnvUtilities::SState>& aArrayOfStates, TUint& aOutputConversionFlags)
   323 {
   324 	TBool returnValue = EFalse;
   325 	TBool foundState = EFalse;
   326 	__ASSERT_DEBUG((aRemainderOfForeign.Length()==0) || (aRemainderOfForeign[0]==KControlCharacterEscape), User::Panic(KPanicReason,EPanicBadRemainderOfForeign));
   327 	if (aRemainderOfForeign.Length()==0)
   328 		{
   329 		return returnValue;
   330 		}
   331 	const TInt numberOfStates=aArrayOfStates.Count();
   332 	TInt i;
   333 	for (i=0; i<numberOfStates; ++i)
   334 		{
   335 		const CnvUtilities::SState& state=aArrayOfStates[i];
   336 		if (MatchesEscapeSequence(aNumberOfForeignBytesConsumed, aHomogeneousRun, aRemainderOfForeign, *state.iEscapeSequence))
   337 			{
   338 			aConversionData=state.iConversionData;
   339 			foundState = ETrue;
   340 			}
   341 		}
   342 	if(!foundState)
   343 		{
   344 		for (i=0; i<numberOfStates; ++i)
   345 			{
   346 			if (IsStartOf(aRemainderOfForeign, *aArrayOfStates[i].iEscapeSequence))
   347 				{
   348 				// aRemainderOfForeign ends with a truncated escape sequence, so ConvertToUnicode cannot convert any more
   349 				aOutputConversionFlags|=CCnvCharacterSetConverter::EOutputConversionFlagInputIsTruncated;
   350 				break;
   351 				}
   352 			}
   353 
   354 		MatchesEscapeSequence(aNumberOfForeignBytesConsumed,aHomogeneousRun,aRemainderOfForeign,aRemainderOfForeign.Left(2));
   355 		aConversionData = &conversionDataDummy;	
   356 		returnValue = ETrue;
   357 		}
   358 		if (aHomogeneousRun.Length()>0)
   359 			{
   360 			returnValue = ETrue;
   361 			}
   362 		return returnValue;		
   363 }
   364 // -----------------------------------------------------------------------------
   365 // ConvertFromUnicodeIntermediateBufferInPlace()
   366 //Default implementation for conversion to the intermediate buffer
   367 //It modifies the unicode buffer before it is converted back to iscii.
   368 //The current implementation of iscii plug-in doesn't require any
   369 //modification to the default implementation
   370 // Returns: Nothing
   371 //
   372 // -----------------------------------------------------------------------------
   373 //
   374 
   375 
   376 LOCAL_C void ConvertFromUnicodeIntermediateBufferInPlace(TInt aStartPositionInDescriptor, TDes8& aDescriptor, TInt& aNumberOfCharactersThatDroppedOut)
   377 {
   378 	CnvUtilities::ConvertFromIntermediateBufferInPlace(aStartPositionInDescriptor, aDescriptor, aNumberOfCharactersThatDroppedOut, KEscapeSequenceDevanagari, 1);
   379 }
   380 
   381 // -----------------------------------------------------------------------------
   382 // DoFindAndModifyBuffer()
   383 //Modifies the iscii buffer by replacing the search buffer with the replacement buffer.
   384 //Introduced for handling multibyte iscii sequence.
   385 //Takes the search buffer array and the replacement buffer arrays as input to it and replaces all
   386 //the occurances of the search buffer with the corresponding replace buffer.
   387 // Returns: Nothing
   388 //
   389 // -----------------------------------------------------------------------------
   390 //
   391 
   392 LOCAL_C void DoFindAndModifyBuffer(TDes8& aModifyBuffer,const TDesC8& aSearchBuffer,const TDesC8& aReplaceBuffer)
   393 {
   394 	FOREVER
   395 	{
   396 		TInt offset;
   397 		__ASSERT_ALWAYS((aSearchBuffer.Length()>= aReplaceBuffer.Length()),User::Panic(KPanicReason,EPanicBadReplacementBuffer));
   398 		if((offset = aModifyBuffer.Find(aSearchBuffer)) != KErrNotFound)
   399 		{
   400 			TUint8 *pointerToBuffer = const_cast<TUint8*> (aModifyBuffer.Ptr());
   401 			Mem::Copy(pointerToBuffer+offset,aReplaceBuffer.Ptr(),aReplaceBuffer.Length());
   402 			Mem::Copy(pointerToBuffer+offset+aReplaceBuffer.Length(),pointerToBuffer+offset+aSearchBuffer.Length(),aModifyBuffer.Length()-aSearchBuffer.Length()-offset);
   403 			aModifyBuffer.SetLength(aModifyBuffer.Length() - aSearchBuffer.Length() + aReplaceBuffer.Length());
   404 		}
   405 		else
   406 			break;
   407 	}
   408 	
   409 }
   410 
   411 // -----------------------------------------------------------------------------
   412 // FindAndModifyBuffer()
   413 //Modifies the iscii buffer by replacing the search buffer with the replacement buffer.
   414 //Calls DoFindAndModifyBuffer() and supplies the search buffer and replacement buffer.
   415 //Introduced for handling multibyte iscii sequence.
   416 // Returns: Nothing
   417 //
   418 // -----------------------------------------------------------------------------
   419 //
   420 
   421 LOCAL_C void FindAndModifyBuffer(TDes8& aModifyBuffer)
   422 {
   423 	TInt ret = KErrNone;
   424     RArray<TPtrC8> searchBuffer;
   425 	RArray<TPtrC8> replaceBuffer;
   426 	
   427 	//If the passed buffer contains the replacement buffer,
   428 	//Then it should not get converted to respective Unicode
   429 	//buffer rather it should get converted to replacement for
   430 	//unconvertible character.
   431 	
   432 	ret |= searchBuffer.Append(KReplacementForExplicitHalant().Right(1));
   433 	ret |= searchBuffer.Append(KReplacementForSoftHalant().Right(1));
   434 	ret |= searchBuffer.Append(KReplacementForOm().Right(1));
   435 	ret |= searchBuffer.Append(KReplacementForAvagraha().Right(1));
   436 	
   437 	ret |= searchBuffer.Append(KReplacementForVocalicRr().Right(1));
   438 	ret |= searchBuffer.Append(KReplacementForVocalicLl().Right(1));
   439 	ret |= searchBuffer.Append(KReplacementForVocalicLSign().Right(1));
   440 	ret |= searchBuffer.Append(KReplacementForVocalicLlSign().Right(1));
   441 	ret |= searchBuffer.Append(KReplacementForVocalicL().Right(1));
   442 	ret |= searchBuffer.Append(KReplacementForVocalicRrSign().Right(1));
   443 	
   444 	//All normal search buffers
   445 	ret |= searchBuffer.Append(KExplicitHalant().Mid(0));
   446 	ret |= searchBuffer.Append(KSoftHalant().Mid(0));
   447 	ret |= searchBuffer.Append(KOm().Mid(0));
   448 	ret |= searchBuffer.Append(KAvagraha().Mid(0));
   449 	
   450 	ret |= searchBuffer.Append(KVocalicRr().Mid(0));
   451 	ret |= searchBuffer.Append(KVocalicLl().Mid(0));
   452 	ret |= searchBuffer.Append(KVocalicLSign().Mid(0));
   453 	ret |= searchBuffer.Append(KVocalicLlSign().Mid(0));
   454 	ret |= searchBuffer.Append(KVocalicL().Mid(0));
   455 	ret |= searchBuffer.Append(KVocalicRrSign().Mid(0));
   456 	
   457 	//The replacement buffer for the odd cases to restrict the 
   458 	//replacement buffers not to get converted to the corresponding 
   459 	//unicode buffer
   460 	
   461 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   462 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   463 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   464 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   465 	
   466 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   467 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   468 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   469 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   470 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   471 	ret |= replaceBuffer.Append(KIsciiUnconvertibleCharacter().Mid(0));
   472 	
   473 	//All normal replace buffers		
   474 	ret |= replaceBuffer.Append(KReplacementForExplicitHalant().Mid(0));
   475 	ret |= replaceBuffer.Append(KReplacementForSoftHalant().Mid(0));
   476 	ret |= replaceBuffer.Append(KReplacementForOm().Mid(0));
   477 	ret |= replaceBuffer.Append(KReplacementForAvagraha().Mid(0));
   478 	
   479 	
   480 	ret |= replaceBuffer.Append(KReplacementForVocalicRr().Mid(0));
   481 	ret |= replaceBuffer.Append(KReplacementForVocalicLl().Mid(0));
   482 	ret |= replaceBuffer.Append(KReplacementForVocalicLSign().Mid(0));
   483 	ret |= replaceBuffer.Append(KReplacementForVocalicLlSign().Mid(0));
   484 	ret |= replaceBuffer.Append(KReplacementForVocalicL().Mid(0));
   485 	ret |= replaceBuffer.Append(KReplacementForVocalicRrSign().Mid(0));
   486 	
   487 	__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
   488 			
   489 	for(TInt counter=0;counter<searchBuffer.Count();counter++)
   490 	{
   491 		DoFindAndModifyBuffer(aModifyBuffer,searchBuffer[counter],replaceBuffer[counter]);
   492 	}
   493 	searchBuffer.Reset();
   494 	replaceBuffer.Reset();
   495 	
   496 }
   497 
   498 // -----------------------------------------------------------------------------
   499 // DoNormalizeReturnValue()
   500 //Modifies the return value(Number of bytes did not get converted) according to the modified
   501 //intermediate buffer. Takes the modified intermediate buffer, return value, search and replace
   502 //buffers. Searches for the replace buffer in the intermediate buffer and if found, it looks for
   503 //the actual buffer corresponding to the replace buffer and adds the difference of the lengths
   504 //of the actual to replace buffer to the return value.
   505 // Returns: Nothing
   506 //
   507 // -----------------------------------------------------------------------------
   508 //
   509 
   510 LOCAL_C void DoNormalizeReturnValue(TUint& aReturnValue,const TDesC8& aBuffer,RArray<TPtrC8>& anArrayOfSearches,RArray<TPtrC8>& anArrayOfReplaces)
   511 {
   512 	TPtr8 buffer(const_cast<TUint8*>(aBuffer.Ptr()),aBuffer.Length(),aBuffer.Length());
   513 	TUint count = anArrayOfSearches.Count();
   514 	FOREVER
   515 	{
   516 		TBool flag = EFalse;
   517 		for(TUint i=0;i<count;i++)
   518 		{
   519 			TPtrC8 searchBufferForComparison(buffer.Right(anArrayOfReplaces[i].Length()));
   520 			TInt returnCompare = searchBufferForComparison.Compare(anArrayOfReplaces[i]);
   521 			if(returnCompare == 0)
   522 			{
   523 				flag =ETrue;
   524 				aReturnValue += (anArrayOfSearches[i].Length() - anArrayOfReplaces[i].Length());
   525 				buffer=buffer.MidTPtr(0,buffer.Length()-anArrayOfReplaces[i].Length());
   526 				break;
   527 			}
   528 		}
   529 		
   530 		if(buffer.Length() == 0)
   531 		{
   532 			break;
   533 		}
   534 		
   535 		if(!flag)
   536 		{
   537 			buffer=buffer.MidTPtr(0,buffer.Length()-1);
   538 		}
   539 	}
   540 
   541 }
   542 
   543 // -----------------------------------------------------------------------------
   544 // NormalizeReturnValue()
   545 //Modifies the return value(Number of bytes did not get converted) according to the 
   546 //replacements done to the iscii buffer before conversion.
   547 //Internally calls DoNormalizeReturnValue() by suppling the search and replacement 
   548 //buffers.
   549 
   550 // Returns: Nothing
   551 //
   552 // -----------------------------------------------------------------------------
   553 //
   554 
   555 LOCAL_C void NormalizeReturnValue(TUint& aReturnValue,const TDesC8& aBuffer)
   556 {
   557 	TInt ret = KErrNone;
   558     RArray<TPtrC8> searchBuffer;
   559 	RArray<TPtrC8> replaceBuffer;
   560 	
   561 	ret |= searchBuffer.Append(KExplicitHalant().Mid(0));
   562 	ret |= searchBuffer.Append(KSoftHalant().Mid(0));
   563 	ret |= searchBuffer.Append(KOm().Mid(0));
   564 	ret |= searchBuffer.Append(KAvagraha().Mid(0));
   565 	
   566 	ret |= searchBuffer.Append(KVocalicRr().Mid(0));
   567 	ret |= searchBuffer.Append(KVocalicLl().Mid(0));
   568 	ret |= searchBuffer.Append(KVocalicLSign().Mid(0));
   569 	ret |= searchBuffer.Append(KVocalicLlSign().Mid(0));
   570 	ret |= searchBuffer.Append(KVocalicL().Mid(0));
   571 	ret |= searchBuffer.Append(KVocalicRrSign().Mid(0));
   572 	
   573 	ret |= replaceBuffer.Append(KReplacementForExplicitHalant().Mid(0));
   574 	ret |= replaceBuffer.Append(KReplacementForSoftHalant().Mid(0));
   575 	ret |= replaceBuffer.Append(KReplacementForOm().Mid(0));
   576 	ret |= replaceBuffer.Append(KReplacementForAvagraha().Mid(0));
   577 	
   578 	ret |= replaceBuffer.Append(KReplacementForVocalicRr().Mid(0));
   579 	ret |= replaceBuffer.Append(KReplacementForVocalicLl().Mid(0));
   580 	ret |= replaceBuffer.Append(KReplacementForVocalicLSign().Mid(0));
   581 	ret |= replaceBuffer.Append(KReplacementForVocalicLlSign().Mid(0));
   582 	ret |= replaceBuffer.Append(KReplacementForVocalicL().Mid(0));
   583 	ret |= replaceBuffer.Append(KReplacementForVocalicRrSign().Mid(0));
   584 	
   585 	__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
   586 	
   587 	DoNormalizeReturnValue(aReturnValue,aBuffer,searchBuffer,replaceBuffer);
   588 	searchBuffer.Reset();
   589 	replaceBuffer.Reset();
   590 }
   591 
   592 // -----------------------------------------------------------------------------
   593 // HandleHomogeneousRun()
   594 //Handles a homogeneous foreign buffer and converts the foreign buffer to unicode
   595 //On return the aUnicode argument contains the converted unicode data.
   596 //Also it sets the return value, returned from the conversion. The return value also
   597 //takes into account if there is any buffer modification done before passing it to
   598 //CCnvCharacterSetConverter::DoConvertToUnicode()
   599 //buffers.
   600 
   601 // Returns: Nothing
   602 //
   603 // -----------------------------------------------------------------------------
   604 //
   605 
   606 LOCAL_C void HandleHomogeneousRun(const SCnvConversionData*& aConversionData, 
   607 							CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters,
   608 							TDes16& aUnicode,
   609 							const TDesC8& aHomogeneousForeign,
   610 							TInt& aNumberOfUnconvertibleCharacters,
   611 							TInt& aIndexOfFirstByteOfFirstUnconvertibleCharacter,
   612 							TUint& aOutputConversionFlags,
   613 							TUint aInputConversionFlags,TInt& aNumberOfForeignBytesConsumed,
   614 							TInt& aReturnValue)
   615 {
   616 	TInt numberOfUnconvertibleCharacters;
   617 	TInt indexOfFirstByteOfFirstUnconvertibleCharacter;
   618 	TUint noOfConsumedBytes = 0;
   619 	if(aConversionData == NULL)
   620 		{
   621 		aReturnValue = CCnvCharacterSetConverter::EErrorIllFormedInput;
   622 		return;
   623 		}
   624 	aReturnValue = CCnvCharacterSetConverter::DoConvertToUnicode(*aConversionData,aDefaultEndiannessOfForeignCharacters,
   625 															  aUnicode,aHomogeneousForeign,numberOfUnconvertibleCharacters,
   626 															  indexOfFirstByteOfFirstUnconvertibleCharacter,aOutputConversionFlags,
   627 
   628 	//The numberOfUnconvertibleCharacters and indexOfFirstByteOfFirstUnconvertibleCharacter are the values with
   629 	//respect to the intermediate iscii buffer and original values aIndexOfFirstByteOfFirstUnconvertibleCharacter and 
   630 	//aNumberOfUnconvertibleCharacters need to be adjusted accordingly.
   631 	
   632 	aInputConversionFlags);
   633 	if(numberOfUnconvertibleCharacters>0)
   634 		{
   635 		if(aNumberOfUnconvertibleCharacters == 0)
   636 			{
   637 			aIndexOfFirstByteOfFirstUnconvertibleCharacter = aNumberOfForeignBytesConsumed + indexOfFirstByteOfFirstUnconvertibleCharacter;
   638 			}
   639 		aNumberOfUnconvertibleCharacters+=numberOfUnconvertibleCharacters;
   640 		}
   641 	noOfConsumedBytes = aHomogeneousForeign.Length();
   642 	//To Check whether it is really required.
   643 	NormalizeReturnValue(noOfConsumedBytes,aHomogeneousForeign);
   644 	aNumberOfForeignBytesConsumed+=noOfConsumedBytes;
   645 	if(aReturnValue>0)
   646 		{
   647 		TUint normalizedReturnValue = aReturnValue;
   648 		
   649 		//There original iscii buffer copied to an intermediate iscii buffer and then modified 
   650 		//and is then passed for conversion. Now, after conversion, the return value needs to
   651 		//be adjusted according to the original buffer. NormalizeReturnValue() does the 
   652 		//same thing.
   653 		
   654 		NormalizeReturnValue(normalizedReturnValue,aHomogeneousForeign);
   655 		aNumberOfForeignBytesConsumed-=normalizedReturnValue;
   656 		aReturnValue=normalizedReturnValue;
   657 		}
   658 	
   659 	//The HandleHomogeneousRun() method is called in a loop and once there is some
   660 	//iscii codes converted to unicode, the ConvertToUnicode() should not return
   661 	//CCnvCharacterSetConverter::EErrorIllFormedInput even though the conversion
   662 	//method does not convert any of the iscii codes ppassed. To ensure that once the
   663 	//first non-zero number of iscii codes are converted, the internal input conversion
   664 	//flag is set to EInputConversionFlagAllowTruncatedInputNotEvenPartlyConsumable.
   665 	
   666 	if(aNumberOfForeignBytesConsumed>0)
   667 		{
   668 		aInputConversionFlags|=CCnvCharacterSetConverter::EInputConversionFlagAllowTruncatedInputNotEvenPartlyConsumable;
   669 		}
   670 	return;
   671 }
   672 
   673 // -----------------------------------------------------------------------------
   674 // IsTruncatedDoubleByteIsciiSequence()
   675 //Checks if anIsciiBuffer is a part of multi byte iscii sequence truncated in the middle.
   676 //If it is a truncated sequence, then returns ETrue else returns EFalse.
   677 //
   678 // Returns: ETrue: If the intermediate input iscii buffer is truncated
   679 //              EFalse: If the intermediate input iscii buffer is not truncated
   680 //
   681 // -----------------------------------------------------------------------------
   682 //
   683 
   684 LOCAL_C TBool IsTruncatedDoubleByteIsciiSequence(const TDesC8& anIsciiBuffer)
   685 {
   686 	RArray<TPtrC8> searchBuffer;
   687 	if(anIsciiBuffer.Length () == 0)
   688 		return EFalse;
   689 	if(anIsciiBuffer[anIsciiBuffer.Length()-1] == 0xEF)
   690 	return ETrue;
   691 	
   692 	TInt appendret = KErrNone;
   693 	appendret |= searchBuffer.Append(KSoftHalant().Mid(0));
   694 	appendret |= searchBuffer.Append(KOm().Mid(0));
   695 	appendret |= searchBuffer.Append(KAvagraha().Mid(0));
   696 	appendret |= searchBuffer.Append(KExplicitHalant().Mid(0));
   697 	__ASSERT_DEBUG(!appendret, User::Panic(_L("RArray append failure"), appendret));
   698 	
   699 	TBool ret = EFalse;
   700 	TBool isNotTruncated =EFalse;
   701 	
   702 	//First check if the intermediate iscii buffer is ending with a complete multi byte sequence.
   703 	//If it ends with a complete multi byte sequence, no need to check if the last character of 
   704 	//intermediate iscii is same as first character of multi byte iscii sequence. And return EFalse.
   705 	for(TUint counter = 0;counter<searchBuffer.Count();counter++)
   706 		{
   707 		if(searchBuffer[counter].Compare(anIsciiBuffer.Right(searchBuffer[counter].Length())) == 0)
   708 			{
   709 			isNotTruncated = ETrue;
   710 			break;
   711 			}
   712 		}
   713 	//If the intermediate iscii buffer is not ending with a complete multi byte sequence, and the 
   714 	//last code of the iscii buffer is a part of the multibyte sequence, then the iscii buffer is a 
   715 	//truncated sequence and in that case return ETrue.
   716 	
   717 	for(TUint counter = 0;counter<searchBuffer.Count();counter++)
   718 		{
   719 		if(isNotTruncated)
   720 			break;
   721 		else if( (anIsciiBuffer[anIsciiBuffer.Length()-1] == searchBuffer[counter][0]))
   722 			{
   723 			ret =ETrue;
   724 			break;
   725 			}
   726 		}
   727 	searchBuffer.Reset();
   728 	return ret;
   729 }
   730 
   731 // -----------------------------------------------------------------------------
   732 // ReplacementForUnconvertibleUnicodeCharacters()
   733 //Returns the replacement character for unconvertible unicode character.
   734 //In the current implementation it is 0x1a (ASCII Substitute character)
   735 //The default implementation calls ReplacementForUnconvertibleUnicodeCharacters_internal()
   736 //in turn which is generated by cnvtool.
   737 //
   738 // Returns: Replacemt for unconvertible unicode characters (0x1a)
   739 //
   740 // -----------------------------------------------------------------------------
   741 //
   742 
   743 const TDesC8& CIsciiImplementation::ReplacementForUnconvertibleUnicodeCharacters()
   744 	{
   745     return ReplacementForUnconvertibleUnicodeCharacters_internal();
   746 	}
   747 
   748 // -----------------------------------------------------------------------------
   749 // ConvertFromUnicode()
   750 //The main conversion function for converting from unicode to iscii
   751 //Loaded and called by the character conversion framework.
   752 //In turn calls CnvUtilities::ConvertFromUnicode() which is Symbian provide
   753 //utility method for converting unicode to modal character codes.
   754 //
   755 // Returns: The numbet of unicode codes it could not convert.
   756 //
   757 // -----------------------------------------------------------------------------
   758 //
   759 
   760 TInt CIsciiImplementation::ConvertFromUnicode(
   761          CCnvCharacterSetConverter::TEndianness 
   762          aDefaultEndiannessOfForeignCharacters, 
   763          const TDesC8& aReplacementForUnconvertibleUnicodeCharacters, 
   764          TDes8& aForeign, 
   765          const TDesC16& aUnicode, 
   766          CCnvCharacterSetConverter::TArrayOfAscendingIndices& aIndicesOfUnconvertibleCharacters )
   767 	{
   768 	TFixedArray<CnvUtilities::SCharacterSet,KNumberOfIndicCharactersetsSupported> aArrayOfCharacterSets;
   769 	aArrayOfCharacterSets[0].iConversionData = &conversionData;
   770 	aArrayOfCharacterSets[0].iConvertFromIntermediateBufferInPlace = ConvertFromUnicodeIntermediateBufferInPlace;
   771 	aArrayOfCharacterSets[0].iEscapeSequence = &KEscapeSequenceDevanagari();
   772 
   773 		return CnvUtilities::ConvertFromUnicode(
   774 				aDefaultEndiannessOfForeignCharacters, 
   775 				aReplacementForUnconvertibleUnicodeCharacters, 
   776 				aForeign, 
   777 				aUnicode, 
   778 				aIndicesOfUnconvertibleCharacters, 
   779 				aArrayOfCharacterSets.Array());
   780 	}
   781 
   782 // -----------------------------------------------------------------------------
   783 // ConvertToUnicode()
   784 //The main conversion function for converting from iscii to unicode
   785 //Loaded and called by the character conversion framework.
   786 //To support some of the iscii characters, the input forign buffer is
   787 //copied to an intermediate buffer and then is then modified and 
   788 //CCnvCharactersetConverter::DoConvertToUnicode() is called with
   789 //the modified buffer. For extensibility of iscii to other Indic languages
   790 //it uses CnvUtilities::SState datastructure. CnvUtilities::SState is a
   791 //Symbian defined class for modal charactersets. The escape sequence 
   792 //is specified to ATR followed by the script selection code and the conversion
   793 //data is specified to be the conversion for the particular script. For the time
   794 //being only Devanagari with script selection code 0x42 is supported. If 
   795 //any of the other script codes are used the conversion leads to unconvertible
   796 //character i.e. 0xFFFD.
   797 // Returns: The numbet of iscii codes it could not convert.
   798 //
   799 // -----------------------------------------------------------------------------
   800 //
   801 
   802 TInt CIsciiImplementation::ConvertToUnicode(
   803          CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters, 
   804          TDes16& aUnicode, 
   805          const TDesC8& aForeign, 
   806          TInt& aState, 
   807          TInt& aNumberOfUnconvertibleCharacters, 
   808          TInt& aIndexOfFirstByteOfFirstUnconvertibleCharacter )
   809 	{
   810 	aNumberOfUnconvertibleCharacters = 0;
   811 	TUint aOutputConversionFlags = CCnvCharacterSetConverter::KStateDefault;
   812 	aIndexOfFirstByteOfFirstUnconvertibleCharacter=-1;
   813 	TUint internalInputConversionFlags = 0;
   814 	TInt numberOfForeignBytesConsumed=0;
   815 	TPtrC8 remainderOfForeign(aForeign);
   816 	TInt returnValue;
   817 	TBool flag = EFalse;
   818 	TBool isSkipMatchSequence = EFalse;
   819 	const SCnvConversionData* convData;
   820 	//Set the iscii conversion data and escape sequence for Devanagari.
   821 	TFixedArray<CnvUtilities::SState,KNumberOfIndicCharactersetsSupported> modals;
   822 	modals[0].iConversionData = &conversionData;
   823 	modals[0].iEscapeSequence = &KEscapeSequenceDevanagari();
   824 
   825 	aUnicode.SetLength(0);
   826 	
   827 	//The internal input conversion flag for conversion is set to CCnvCharacterSetConverter::EInputConversionFlagAppend
   828 	//so that for each conversion in the conversion loop, the generated conversion buffer is appened to the aUnicode buffer.
   829 	internalInputConversionFlags |= CCnvCharacterSetConverter::EInputConversionFlagAppend;
   830 	if (aForeign.Length()==0)
   831 	{
   832 		return 0;
   833 	}
   834 	//Get the conversion data from the previous call else the conversion data is set to the default 
   835 	//conversion data, i.e. Devanagari.
   836 	convData=(aState!=CCnvCharacterSetConverter::KStateDefault)? REINTERPRET_CAST(const SCnvConversionData*, aState): modals[0].iConversionData;
   837 	FOREVER
   838 	{
   839 		TBuf8<KMaximumLengthOfIntermediateBuffer> intermediateBuffer;
   840 		TUint numberOfForeignBytesConsumedThisTime = 0;
   841 		if((remainderOfForeign.Length() >=KMaximumLengthOfIntermediateBuffer) && (aUnicode.MaxLength()-aUnicode.Length() >=KMaximumLengthOfIntermediateBuffer))
   842 		{
   843 			numberOfForeignBytesConsumedThisTime = KMaximumLengthOfIntermediateBuffer;
   844 			intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
   845 			//If the intermediate buffer is a part of truncated buffer sequence but the
   846 			//actual input buffer is not truncated then truncated sequence is not converted.
   847 			//The intermediate buffer is modified so as not to contain the truncated sequence.
   848 			
   849 			flag = IsTruncatedDoubleByteIsciiSequence(intermediateBuffer);
   850 			if(flag)
   851 				{
   852 					numberOfForeignBytesConsumedThisTime --;
   853 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);					
   854 				}
   855 			
   856 		}
   857 		else
   858 		{
   859 			flag = IsTruncatedDoubleByteIsciiSequence(remainderOfForeign.Left(aUnicode.MaxLength()-aUnicode.Length()));
   860 			if(!flag)
   861 				{
   862 				numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length();
   863 				intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
   864 				}
   865 			else
   866 				{
   867 				if(aForeign.Length()>(numberOfForeignBytesConsumed+aUnicode.Length()))
   868 					{
   869 					numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length()-1;
   870 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
   871 					break;
   872 					}
   873 				else
   874 					{
   875 					numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length();
   876 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
   877 					}
   878 				}
   879 		}			
   880 		
   881 		//The input intermediate iscii buffer is modified with the search and replace
   882 		//buffers. It is required for supporting multibyte iscii sequences.		
   883 		FindAndModifyBuffer(intermediateBuffer);
   884 		TPtrC8 remainderOfForeignInternal(intermediateBuffer);
   885 		TPtrC8 homogeneousRun;
   886 		const TInt startOfNextEscapeSequence=intermediateBuffer.Locate(KControlCharacterEscape);
   887 		if (startOfNextEscapeSequence!=0) 
   888 			{
   889 			if (startOfNextEscapeSequence==KErrNotFound)
   890 				{
   891 				homogeneousRun.Set(remainderOfForeignInternal);
   892 				remainderOfForeignInternal.Set(NULL, 0);
   893 				}
   894 			else
   895 				{
   896 				__ASSERT_DEBUG(startOfNextEscapeSequence>0, User::Panic(KPanicReason,EPanicBadStartOfNextEscapeSequence));
   897 				homogeneousRun.Set(remainderOfForeignInternal.Left(startOfNextEscapeSequence));
   898 				remainderOfForeignInternal.Set(remainderOfForeignInternal.Mid(startOfNextEscapeSequence));
   899 				}
   900 			isSkipMatchSequence = ETrue;
   901 			}
   902 		FOREVER
   903 			{
   904 			if(!isSkipMatchSequence)
   905 				{
   906 				if (!NextHomogeneousForeignRun(convData, numberOfForeignBytesConsumed, homogeneousRun, 
   907 											remainderOfForeignInternal, modals.Array(), aOutputConversionFlags))
   908 					{
   909 					break;
   910 					}
   911 				}
   912 			HandleHomogeneousRun( convData, aDefaultEndiannessOfForeignCharacters, aUnicode, homogeneousRun, aNumberOfUnconvertibleCharacters, 
   913 								aIndexOfFirstByteOfFirstUnconvertibleCharacter,aOutputConversionFlags,internalInputConversionFlags,
   914 								numberOfForeignBytesConsumed,returnValue);
   915 			if(returnValue<0)
   916 				{
   917 				return returnValue;
   918 				}
   919 			isSkipMatchSequence = EFalse;
   920 			}
   921 		if(returnValue > 0)
   922 			break;
   923 		if ((!flag && (numberOfForeignBytesConsumedThisTime != KMaximumLengthOfIntermediateBuffer)) || (flag && (numberOfForeignBytesConsumedThisTime != (KMaximumLengthOfIntermediateBuffer-1) )))
   924 			break;
   925 		remainderOfForeign.Set(aForeign.Mid(numberOfForeignBytesConsumed));				
   926 	}
   927 	//If the number of iscii bytes consumed by the conversion is zero also the output conversion
   928 	//flag is not set to EOutputConversionFlagInputIsTruncated, then return EErrorIllFormedInput.
   929 	if ((numberOfForeignBytesConsumed==0) && (aOutputConversionFlags&CCnvCharacterSetConverter::EOutputConversionFlagInputIsTruncated))
   930 		{
   931 		return CCnvCharacterSetConverter::EErrorIllFormedInput;
   932 		}
   933 	//Set the conversion data sothat next time when ConvertToUnicode() is called,
   934 	//will use the previous conversion data.
   935 	aState=REINTERPRET_CAST(TInt, convData);
   936 	return aForeign.Length()-numberOfForeignBytesConsumed;
   937     }
   938 
   939 // -----------------------------------------------------------------------------
   940 // IsInThisCharacterSetL()
   941 //The method tells how probable it is that a sample piece of text is encoded in this character set.
   942 //On return aConfidenceLevel, indicates how confident the function is about its return value. For
   943 //iscii it is the default implementation and it does not implement the autodetect functionality.
   944 //Loaded and called by the character conversion framework.
   945 //
   946 // Returns: EFalse: To tell CCnvCharacterSetConverter::AutoDetectCharacterSetL()  that the plug-in DLL
   947 //                          is not implementing a function of this signature and is therefore empty.
   948 //
   949 // -----------------------------------------------------------------------------
   950 //
   951 
   952 
   953 //Default implementation for IsInThisCharacterSetL()
   954 
   955 TBool CIsciiImplementation::IsInThisCharacterSetL(
   956          TBool& aSetToTrue, 
   957          TInt& aConfidenceLevel, 
   958          const TDesC8& )
   959 	{
   960     	aSetToTrue = EFalse;
   961 	aConfidenceLevel = 0;
   962 	return EFalse;
   963 	}
   964 
   965 // -----------------------------------------------------------------------------
   966 // NewL()
   967 //Factory function for CIsciiImplementation(). Instantiates a CIsciiImplementation object on heap
   968 //and returns the pointer to it.
   969 //
   970 // Returns: CIsciiImplementation*
   971 //
   972 // -----------------------------------------------------------------------------
   973 //
   974 	
   975 CIsciiImplementation* CIsciiImplementation::NewL()
   976     {
   977     CIsciiImplementation* self = new(ELeave) CIsciiImplementation;
   978     return self;
   979     }
   980 
   981 // -----------------------------------------------------------------------------
   982 // CIsciiImplementation()
   983 //default constructor, does nothing
   984 //
   985 // Returns: Nothing
   986 //
   987 // -----------------------------------------------------------------------------
   988 //
   989 CIsciiImplementation::CIsciiImplementation()
   990     {
   991     }
   992 
   993 // -----------------------------------------------------------------------------
   994 // ~CIsciiImplementation()
   995 //default desstructor, does nothing
   996 //
   997 // Returns: Nothing
   998 //
   999 // -----------------------------------------------------------------------------
  1000 //
  1001 
  1002 CIsciiImplementation::~CIsciiImplementation()
  1003     {
  1004     }
  1005 
  1006 // ECOM CREATION FUNCTION
  1007 const TImplementationProxy ImplementationTable[] = 
  1008     {
  1009     // Used also in 0x1027508E.rss ( implementation_uid )
  1010     IMPLEMENTATION_PROXY_ENTRY( 0x1027508E, CIsciiImplementation::NewL )
  1011     };
  1012 
  1013 // -----------------------------------------------------------------------------
  1014 // ImplementationGroupProxy()
  1015 //Returns a pointer to TImplementationProxy object which contains the implementation uid vs factory
  1016 //function table. Also on return sets the aTableCount to the number of entries in the table.
  1017 //
  1018 // Returns: TImplementationProxy*
  1019 //
  1020 // -----------------------------------------------------------------------------
  1021 //
  1022 EXPORT_C const TImplementationProxy* ImplementationGroupProxy( TInt& aTableCount )
  1023     {
  1024     aTableCount = sizeof( ImplementationTable ) / sizeof(TImplementationProxy);
  1025     return ImplementationTable;
  1026     }
  1027 #else
  1028 
  1029 #include <convplug.h>
  1030 
  1031 #ifndef EKA2
  1032 // -----------------------------------------------------------------------------
  1033 // E32Dll()
  1034 //For EKA1 this is the entry point for the DLL.
  1035 //
  1036 // Returns: KErrNone
  1037 //
  1038 // -----------------------------------------------------------------------------
  1039 //
  1040 GLDEF_C TInt E32Dll(TDllReason)
  1041 {
  1042 	return KErrNone;
  1043 }
  1044 #endif
  1045 
  1046 //Checks if a descriptor starts with another descriptor at the begining.
  1047 LOCAL_C TBool IsStartOf(const TDesC8& aEscapeSequence, const TDesC8& aBuffer)
  1048 {
  1049 	const TInt lengthOfStart=aEscapeSequence.Length();
  1050 	return (aBuffer.Length()>=lengthOfStart) && (aBuffer.Left(lengthOfStart)==aEscapeSequence);
  1051 }
  1052 // -----------------------------------------------------------------------------
  1053 // MatchesEscapeSequence()
  1054 //If the remainder of the foreign text starts with the passed escapesequence, modifies the remainder of the foreign text
  1055 //and  sets the homogeneous run buffer that uses the same conversion data.
  1056 //The homogeneous run buffer is the buffer that will use a single conversion data, it doesn't contain the attribute code
  1057 // neither it contains the script switching code.
  1058 //The aRemainderOfForeign buffer 
  1059 
  1060 // Returns: ETrue: If the sequence contains the escape sequence
  1061 //              EFalse: If the sequence does not contain the escape sequence
  1062 //
  1063 // -----------------------------------------------------------------------------
  1064 //
  1065 LOCAL_C TBool MatchesEscapeSequence(TInt& aNumberOfForeignBytesConsumed, TPtrC8& aHomogeneousRun,
  1066 		TPtrC8& aRemainderOfForeign, const TDesC8& aEscapeSequence)
  1067 {
  1068 	const TInt lengthOfEscapeSequence=aEscapeSequence.Length();
  1069 	if (IsStartOf(aEscapeSequence, aRemainderOfForeign))
  1070 		{
  1071 		aRemainderOfForeign.Set(aRemainderOfForeign.Mid(lengthOfEscapeSequence));
  1072 		const TInt startOfNextEscapeSequence=aRemainderOfForeign.Locate(KControlCharacterEscape);
  1073 		if (startOfNextEscapeSequence==KErrNotFound)
  1074 			{
  1075 			aHomogeneousRun.Set(aRemainderOfForeign);
  1076 			aRemainderOfForeign.Set(NULL, 0);
  1077 			}
  1078 		else
  1079 			{
  1080 			aHomogeneousRun.Set(aRemainderOfForeign.Left(startOfNextEscapeSequence));
  1081 			aRemainderOfForeign.Set(aRemainderOfForeign.Mid(startOfNextEscapeSequence));
  1082 			}
  1083 		aNumberOfForeignBytesConsumed+=lengthOfEscapeSequence;
  1084 		return ETrue;
  1085 		}
  1086 	return EFalse;
  1087 }
  1088 
  1089 // -----------------------------------------------------------------------------
  1090 // NextHomogeneousForeignRun()
  1091 //Matches the escape sequence of each of the elements of the SState array with the remainder of
  1092 //foreign text and if the escape sequence matches with the start of remainder of the foreign text, 
  1093 //then the conversion data is set to the conversion data corresponding to the escape sequence
  1094 //Also the homogeneous foreign text for conversion with the same escape sequence is set.
  1095 
  1096 // Returns: ETrue: If length of the remainder of foreign buffer is nonzero.
  1097 //              EFalse: If length of the remainder of foreign buffer is zero.
  1098 //
  1099 // -----------------------------------------------------------------------------
  1100 //
  1101 
  1102 
  1103 LOCAL_C TBool NextHomogeneousForeignRun(const SCnvConversionData*& aConversionData, TInt& aNumberOfForeignBytesConsumed, TPtrC8& aHomogeneousRun, TPtrC8& aRemainderOfForeign, const TArray<CnvUtilities::SState>& aArrayOfStates, TUint& aOutputConversionFlags)
  1104 {
  1105 	TBool returnValue = EFalse;
  1106 	TBool foundState = EFalse;
  1107 	__ASSERT_DEBUG((aRemainderOfForeign.Length()==0) || (aRemainderOfForeign[0]==KControlCharacterEscape), User::Panic(KPanicReason,EPanicBadRemainderOfForeign));
  1108 	if (aRemainderOfForeign.Length()==0)
  1109 		{
  1110 		return returnValue;
  1111 		}
  1112 	const TInt numberOfStates=aArrayOfStates.Count();
  1113 	TInt i;
  1114 	for (i=0; i<numberOfStates; ++i)
  1115 		{
  1116 		const CnvUtilities::SState& state=aArrayOfStates[i];
  1117 		if (MatchesEscapeSequence(aNumberOfForeignBytesConsumed, aHomogeneousRun, aRemainderOfForeign, *state.iEscapeSequence))
  1118 			{
  1119 			aConversionData=state.iConversionData;
  1120 			foundState = ETrue;
  1121 			}
  1122 		}
  1123 	if(!foundState)
  1124 		{
  1125 		for (i=0; i<numberOfStates; ++i)
  1126 			{
  1127 			if (IsStartOf(aRemainderOfForeign, *aArrayOfStates[i].iEscapeSequence))
  1128 				{
  1129 				// aRemainderOfForeign ends with a truncated escape sequence, so ConvertToUnicode cannot convert any more
  1130 				aOutputConversionFlags|=CCnvCharacterSetConverter::EOutputConversionFlagInputIsTruncated;
  1131 				break;
  1132 				}
  1133 			}
  1134 
  1135 		MatchesEscapeSequence(aNumberOfForeignBytesConsumed,aHomogeneousRun,aRemainderOfForeign,aRemainderOfForeign.Left(2));
  1136 		aConversionData = &conversionDataDummy;	
  1137 		returnValue = ETrue;
  1138 		}
  1139 		if (aHomogeneousRun.Length()>0)
  1140 			{
  1141 			returnValue = ETrue;
  1142 			}
  1143 		return returnValue;		
  1144 }
  1145 // -----------------------------------------------------------------------------
  1146 // ConvertFromUnicodeIntermediateBufferInPlace()
  1147 //Default implementation for conversion to the intermediate buffer
  1148 //It modifies the unicode buffer before it is converted back to iscii.
  1149 //The current implementation of iscii plug-in doesn't require any
  1150 //modification to the default implementation
  1151 // Returns: Nothing
  1152 //
  1153 // -----------------------------------------------------------------------------
  1154 //
  1155 
  1156 
  1157 LOCAL_C void ConvertFromUnicodeIntermediateBufferInPlace(TInt aStartPositionInDescriptor, TDes8& aDescriptor, TInt& aNumberOfCharactersThatDroppedOut)
  1158 {
  1159 	CnvUtilities::ConvertFromIntermediateBufferInPlace(aStartPositionInDescriptor, aDescriptor, aNumberOfCharactersThatDroppedOut, KEscapeSequenceDevanagari, 1);
  1160 }
  1161 
  1162 // -----------------------------------------------------------------------------
  1163 // DoFindAndModifyBuffer()
  1164 //Modifies the iscii buffer by replacing the search buffer with the replacement buffer.
  1165 //Introduced for handling multibyte iscii sequence.
  1166 //Takes the search buffer array and the replacement buffer arrays as input to it and replaces all
  1167 //the occurances of the search buffer with the corresponding replace buffer.
  1168 // Returns: Nothing
  1169 //
  1170 // -----------------------------------------------------------------------------
  1171 //
  1172 
  1173 LOCAL_C void DoFindAndModifyBuffer(TDes8& aModifyBuffer,const TDesC8& aSearchBuffer,const TDesC8& aReplaceBuffer)
  1174 {
  1175 	FOREVER
  1176 	{
  1177 		TInt offset;
  1178 		__ASSERT_ALWAYS((aSearchBuffer.Length()>= aReplaceBuffer.Length()),User::Panic(KPanicReason,EPanicBadReplacementBuffer));
  1179 		if((offset = aModifyBuffer.Find(aSearchBuffer)) != KErrNotFound)
  1180 		{
  1181 			TUint8 *pointerToBuffer = const_cast<TUint8*> (aModifyBuffer.Ptr());
  1182 			Mem::Copy(pointerToBuffer+offset,aReplaceBuffer.Ptr(),aReplaceBuffer.Length());
  1183 			Mem::Copy(pointerToBuffer+offset+aReplaceBuffer.Length(),pointerToBuffer+offset+aSearchBuffer.Length(),aModifyBuffer.Length()-aSearchBuffer.Length()-offset);
  1184 			aModifyBuffer.SetLength(aModifyBuffer.Length() - aSearchBuffer.Length() + aReplaceBuffer.Length());
  1185 		}
  1186 		else
  1187 			break;
  1188 	}
  1189 	
  1190 }
  1191 
  1192 // -----------------------------------------------------------------------------
  1193 // FindAndModifyBuffer()
  1194 //Modifies the iscii buffer by replacing the search buffer with the replacement buffer.
  1195 //Calls DoFindAndModifyBuffer() and supplies the search buffer and replacement buffer.
  1196 //Introduced for handling multibyte iscii sequence.
  1197 // Returns: Nothing
  1198 //
  1199 // -----------------------------------------------------------------------------
  1200 //
  1201 
  1202 LOCAL_C void FindAndModifyBuffer(TDes8& aModifyBuffer)
  1203 {
  1204 	RArray<TPtrC8> searchBuffer;
  1205 	RArray<TPtrC8> replaceBuffer;
  1206 	
  1207 	TInt ret = KErrNone;
  1208 	ret |= searchBuffer.Append(KExplicitHalant().Mid(0));
  1209 	ret |= searchBuffer.Append(KSoftHalant().Mid(0));
  1210 	ret |= searchBuffer.Append(KOm().Mid(0));
  1211 	ret |= searchBuffer.Append(KAvagraha().Mid(0));
  1212 	
  1213 	ret |= replaceBuffer.Append(KReplacementForExplicitHalant().Mid(0));
  1214 	ret |= replaceBuffer.Append(KReplacementForSoftHalant().Mid(0));
  1215 	ret |= replaceBuffer.Append(KReplacementForOm().Mid(0));
  1216 	ret |= replaceBuffer.Append(KReplacementForAvagraha().Mid(0));
  1217 	
  1218 	__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
  1219 	
  1220 	for(TInt counter=0;counter<searchBuffer.Count();counter++)
  1221 	{
  1222 		DoFindAndModifyBuffer(aModifyBuffer,searchBuffer[counter],replaceBuffer[counter]);
  1223 	}
  1224 	searchBuffer.Reset();
  1225 	replaceBuffer.Reset();
  1226 	
  1227 }
  1228 
  1229 // -----------------------------------------------------------------------------
  1230 // DoNormalizeReturnValue()
  1231 //Modifies the return value(Number of bytes did not get converted) according to the modified
  1232 //intermediate buffer. Takes the modified intermediate buffer, return value, search and replace
  1233 //buffers. Searches for the replace buffer in the intermediate buffer and if found, it looks for
  1234 //the actual buffer corresponding to the replace buffer and adds the difference of the lengths
  1235 //of the actual to replace buffer to the return value.
  1236 // Returns: Nothing
  1237 //
  1238 // -----------------------------------------------------------------------------
  1239 //
  1240 
  1241 LOCAL_C void DoNormalizeReturnValue(TUint& aReturnValue,const TDesC8& aBuffer,RArray<TPtrC8>& anArrayOfSearches,RArray<TPtrC8>& anArrayOfReplaces)
  1242 {
  1243 	TPtr8 buffer(const_cast<TUint8*>(aBuffer.Ptr()),aBuffer.Length(),aBuffer.Length());
  1244 	TUint count = anArrayOfSearches.Count();
  1245 	FOREVER
  1246 	{
  1247 		TBool flag = EFalse;
  1248 		for(TUint i=0;i<count;i++)
  1249 		{
  1250 			TPtrC8 searchBufferForComparison(buffer.Right(anArrayOfReplaces[i].Length()));
  1251 			TInt returnCompare = searchBufferForComparison.Compare(anArrayOfReplaces[i]);
  1252 			if(returnCompare == 0)
  1253 			{
  1254 				flag =ETrue;
  1255 				aReturnValue += (anArrayOfSearches[i].Length() - anArrayOfReplaces[i].Length());
  1256 				buffer=buffer.MidTPtr(0,buffer.Length()-anArrayOfReplaces[i].Length());
  1257 				break;
  1258 			}
  1259 		}
  1260 		
  1261 		if(buffer.Length() == 0)
  1262 		{
  1263 			break;
  1264 		}
  1265 		
  1266 		if(!flag)
  1267 		{
  1268 			buffer=buffer.MidTPtr(0,buffer.Length()-1);
  1269 		}
  1270 	}
  1271 
  1272 }
  1273 
  1274 // -----------------------------------------------------------------------------
  1275 // NormalizeReturnValue()
  1276 //Modifies the return value(Number of bytes did not get converted) according to the 
  1277 //replacements done to the iscii buffer before conversion.
  1278 //Internally calls DoNormalizeReturnValue() by suppling the search and replacement 
  1279 //buffers.
  1280 
  1281 // Returns: Nothing
  1282 //
  1283 // -----------------------------------------------------------------------------
  1284 //
  1285 
  1286 LOCAL_C void NormalizeReturnValue(TUint& aReturnValue,const TDesC8& aBuffer)
  1287 {
  1288 	RArray<TPtrC8> searchBuffer;
  1289 	RArray<TPtrC8> replaceBuffer;
  1290 	TInt ret =KErrNone;
  1291 	ret |= searchBuffer.Append(KExplicitHalant().Mid(0));
  1292 	ret |= searchBuffer.Append(KSoftHalant().Mid(0));
  1293 	ret |= searchBuffer.Append(KOm().Mid(0));
  1294 	ret |= searchBuffer.Append(KAvagraha().Mid(0));
  1295 	
  1296 	ret |= replaceBuffer.Append(KReplacementForExplicitHalant().Mid(0));
  1297 	ret |= replaceBuffer.Append(KReplacementForSoftHalant().Mid(0));
  1298 	ret |= replaceBuffer.Append(KReplacementForOm().Mid(0));
  1299 	ret |= replaceBuffer.Append(KReplacementForAvagraha().Mid(0));
  1300 	
  1301 	__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
  1302 	
  1303 	DoNormalizeReturnValue(aReturnValue,aBuffer,searchBuffer,replaceBuffer);
  1304 	searchBuffer.Reset();
  1305 	replaceBuffer.Reset();
  1306 }
  1307 
  1308 // -----------------------------------------------------------------------------
  1309 // HandleHomogeneousRun()
  1310 //Handles a homogeneous foreign buffer and converts the foreign buffer to unicode
  1311 //On return the aUnicode argument contains the converted unicode data.
  1312 //Also it sets the return value, returned from the conversion. The return value also
  1313 //takes into account if there is any buffer modification done before passing it to
  1314 //CCnvCharacterSetConverter::DoConvertToUnicode()
  1315 //buffers.
  1316 
  1317 // Returns: Nothing
  1318 //
  1319 // -----------------------------------------------------------------------------
  1320 //
  1321 
  1322 LOCAL_C void HandleHomogeneousRun(const SCnvConversionData*& aConversionData, 
  1323 							CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters,
  1324 							TDes16& aUnicode,
  1325 							const TDesC8& aHomogeneousForeign,
  1326 							TInt& aNumberOfUnconvertibleCharacters,
  1327 							TInt& aIndexOfFirstByteOfFirstUnconvertibleCharacter,
  1328 							TUint& aOutputConversionFlags,
  1329 							TUint aInputConversionFlags,TInt& aNumberOfForeignBytesConsumed,
  1330 							TInt& aReturnValue)
  1331 {
  1332 	TInt numberOfUnconvertibleCharacters;
  1333 	TInt indexOfFirstByteOfFirstUnconvertibleCharacter;
  1334 	TUint noOfConsumedBytes = 0;
  1335 	if(aConversionData == NULL)
  1336 		{
  1337 		aReturnValue = CCnvCharacterSetConverter::EErrorIllFormedInput;
  1338 		return;
  1339 		}
  1340 	aReturnValue = CCnvCharacterSetConverter::DoConvertToUnicode(*aConversionData,aDefaultEndiannessOfForeignCharacters,
  1341 															  aUnicode,aHomogeneousForeign,numberOfUnconvertibleCharacters,
  1342 															  indexOfFirstByteOfFirstUnconvertibleCharacter,aOutputConversionFlags,
  1343 
  1344 	//The numberOfUnconvertibleCharacters and indexOfFirstByteOfFirstUnconvertibleCharacter are the values with
  1345 	//respect to the intermediate iscii buffer and original values aIndexOfFirstByteOfFirstUnconvertibleCharacter and 
  1346 	//aNumberOfUnconvertibleCharacters need to be adjusted accordingly.
  1347 	
  1348 	aInputConversionFlags);
  1349 	if(numberOfUnconvertibleCharacters>0)
  1350 		{
  1351 		if(aNumberOfUnconvertibleCharacters == 0)
  1352 			{
  1353 			aIndexOfFirstByteOfFirstUnconvertibleCharacter = aNumberOfForeignBytesConsumed + indexOfFirstByteOfFirstUnconvertibleCharacter;
  1354 			}
  1355 		aNumberOfUnconvertibleCharacters+=numberOfUnconvertibleCharacters;
  1356 		}
  1357 	noOfConsumedBytes = aHomogeneousForeign.Length();
  1358 	//To Check whether it is really required.
  1359 	NormalizeReturnValue(noOfConsumedBytes,aHomogeneousForeign);
  1360 	aNumberOfForeignBytesConsumed+=noOfConsumedBytes;
  1361 	if(aReturnValue>0)
  1362 		{
  1363 		TUint normalizedReturnValue = aReturnValue;
  1364 		
  1365 		//There original iscii buffer copied to an intermediate iscii buffer and then modified 
  1366 		//and is then passed for conversion. Now, after conversion, the return value needs to
  1367 		//be adjusted according to the original buffer. NormalizeReturnValue() does the 
  1368 		//same thing.
  1369 		
  1370 		NormalizeReturnValue(normalizedReturnValue,aHomogeneousForeign);
  1371 		aNumberOfForeignBytesConsumed-=normalizedReturnValue;
  1372 		aReturnValue=normalizedReturnValue;
  1373 		}
  1374 	
  1375 	//The HandleHomogeneousRun() method is called in a loop and once there is some
  1376 	//iscii codes converted to unicode, the ConvertToUnicode() should not return
  1377 	//CCnvCharacterSetConverter::EErrorIllFormedInput even though the conversion
  1378 	//method does not convert any of the iscii codes ppassed. To ensure that once the
  1379 	//first non-zero number of iscii codes are converted, the internal input conversion
  1380 	//flag is set to EInputConversionFlagAllowTruncatedInputNotEvenPartlyConsumable.
  1381 	
  1382 	if(aNumberOfForeignBytesConsumed>0)
  1383 		{
  1384 		aInputConversionFlags|=CCnvCharacterSetConverter::EInputConversionFlagAllowTruncatedInputNotEvenPartlyConsumable;
  1385 		}
  1386 	return;
  1387 }
  1388 
  1389 // -----------------------------------------------------------------------------
  1390 // IsTruncatedDoubleByteIsciiSequence()
  1391 //Checks if anIsciiBuffer is a part of multi byte iscii sequence truncated in the middle.
  1392 //If it is a truncated sequence, then returns ETrue else returns EFalse.
  1393 //
  1394 // Returns: ETrue: If the intermediate input iscii buffer is truncated
  1395 //              EFalse: If the intermediate input iscii buffer is not truncated
  1396 //
  1397 // -----------------------------------------------------------------------------
  1398 //
  1399 
  1400 LOCAL_C TBool IsTruncatedDoubleByteIsciiSequence(const TDesC8& anIsciiBuffer)
  1401 {
  1402 	RArray<TPtrC8> searchBuffer;
  1403 	if(anIsciiBuffer.Length () == 0)
  1404 		return EFalse;
  1405 	if(anIsciiBuffer[anIsciiBuffer.Length()-1] == 0xEF)
  1406 	return ETrue;
  1407 	
  1408 	TInt appendret = KErrNone;
  1409 	appendret |= searchBuffer.Append(KSoftHalant().Mid(0));
  1410 	appendret |= searchBuffer.Append(KOm().Mid(0));
  1411 	appendret |= searchBuffer.Append(KAvagraha().Mid(0));
  1412 	appendret |= searchBuffer.Append(KExplicitHalant().Mid(0));
  1413 	__ASSERT_DEBUG(!ret, User::Panic(_L("RArray append failure"), ret));
  1414 	
  1415 	TBool ret = EFalse;
  1416 	TBool isNotTruncated =EFalse;
  1417 	
  1418 	//First check if the intermediate iscii buffer is ending with a complete multi byte sequence.
  1419 	//If it ends with a complete multi byte sequence, no need to check if the last character of 
  1420 	//intermediate iscii is same as first character of multi byte iscii sequence. And return EFalse.
  1421 	for(TUint counter = 0;counter<searchBuffer.Count();counter++)
  1422 		{
  1423 		if(searchBuffer[counter].Compare(anIsciiBuffer.Right(searchBuffer[counter].Length())) == 0)
  1424 			{
  1425 			isNotTruncated = ETrue;
  1426 			break;
  1427 			}
  1428 		}
  1429 	//If the intermediate iscii buffer is not ending with a complete multi byte sequence, and the 
  1430 	//last code of the iscii buffer is a part of the multibyte sequence, then the iscii buffer is a 
  1431 	//truncated sequence and in that case return ETrue.
  1432 	
  1433 	for(TUint counter = 0;counter<searchBuffer.Count();counter++)
  1434 		{
  1435 		if(isNotTruncated)
  1436 			break;
  1437 		else if( (anIsciiBuffer[anIsciiBuffer.Length()-1] == searchBuffer[counter][0]))
  1438 			{
  1439 			ret =ETrue;
  1440 			break;
  1441 			}
  1442 		}
  1443 	searchBuffer.Reset();
  1444 	return ret;
  1445 }
  1446 
  1447 // -----------------------------------------------------------------------------
  1448 // ReplacementForUnconvertibleUnicodeCharacters()
  1449 //Returns the replacement character for unconvertible unicode character.
  1450 //In the current implementation it is 0x1a (ASCII Substitute character)
  1451 //The default implementation calls ReplacementForUnconvertibleUnicodeCharacters_internal()
  1452 //in turn which is generated by cnvtool.
  1453 //
  1454 // Returns: Replacemt for unconvertible unicode characters (0x1a)
  1455 //
  1456 // -----------------------------------------------------------------------------
  1457 //
  1458 
  1459 EXPORT_C const TDesC8& ReplacementForUnconvertibleUnicodeCharacters()
  1460 	{
  1461     return ReplacementForUnconvertibleUnicodeCharacters_internal();
  1462 	}
  1463 
  1464 // -----------------------------------------------------------------------------
  1465 // ConvertFromUnicode()
  1466 //The main conversion function for converting from unicode to iscii
  1467 //Loaded and called by the character conversion framework.
  1468 //In turn calls CnvUtilities::ConvertFromUnicode() which is Symbian provide
  1469 //utility method for converting unicode to modal character codes.
  1470 //
  1471 // Returns: The numbet of unicode codes it could not convert.
  1472 //
  1473 // -----------------------------------------------------------------------------
  1474 //
  1475 
  1476 EXPORT_C TInt ConvertFromUnicode(
  1477          CCnvCharacterSetConverter::TEndianness 
  1478          aDefaultEndiannessOfForeignCharacters, 
  1479          const TDesC8& aReplacementForUnconvertibleUnicodeCharacters, 
  1480          TDes8& aForeign, 
  1481          const TDesC16& aUnicode, 
  1482          CCnvCharacterSetConverter::TArrayOfAscendingIndices& aIndicesOfUnconvertibleCharacters )
  1483 	{
  1484 	TFixedArray<CnvUtilities::SCharacterSet,KNumberOfIndicCharactersetsSupported> aArrayOfCharacterSets;
  1485 	aArrayOfCharacterSets[0].iConversionData = &conversionData;
  1486 	aArrayOfCharacterSets[0].iConvertFromIntermediateBufferInPlace = ConvertFromUnicodeIntermediateBufferInPlace;
  1487 	aArrayOfCharacterSets[0].iEscapeSequence = &KEscapeSequenceDevanagari();
  1488 
  1489 		return CnvUtilities::ConvertFromUnicode(
  1490 				aDefaultEndiannessOfForeignCharacters, 
  1491 				aReplacementForUnconvertibleUnicodeCharacters, 
  1492 				aForeign, 
  1493 				aUnicode, 
  1494 				aIndicesOfUnconvertibleCharacters, 
  1495 				aArrayOfCharacterSets.Array());
  1496 	}
  1497 
  1498 // -----------------------------------------------------------------------------
  1499 // ConvertToUnicode()
  1500 //The main conversion function for converting from iscii to unicode
  1501 //Loaded and called by the character conversion framework.
  1502 //To support some of the iscii characters, the input forign buffer is
  1503 //copied to an intermediate buffer and then is then modified and 
  1504 //CCnvCharactersetConverter::DoConvertToUnicode() is called with
  1505 //the modified buffer. For extensibility of iscii to other Indic languages
  1506 //it uses CnvUtilities::SState datastructure. CnvUtilities::SState is a
  1507 //Symbian defined class for modal charactersets. The escape sequence 
  1508 //is specified to ATR followed by the script selection code and the conversion
  1509 //data is specified to be the conversion for the particular script. For the time
  1510 //being only Devanagari with script selection code 0x42 is supported. If 
  1511 //any of the other script codes are used the conversion leads to unconvertible
  1512 //character i.e. 0xFFFD.
  1513 // Returns: The numbet of iscii codes it could not convert.
  1514 //
  1515 // -----------------------------------------------------------------------------
  1516 //
  1517 
  1518 EXPORT_C TInt ConvertToUnicode(
  1519          CCnvCharacterSetConverter::TEndianness aDefaultEndiannessOfForeignCharacters, 
  1520          TDes16& aUnicode, 
  1521          const TDesC8& aForeign, 
  1522          TInt& aState, 
  1523          TInt& aNumberOfUnconvertibleCharacters, 
  1524          TInt& aIndexOfFirstByteOfFirstUnconvertibleCharacter )
  1525 	{
  1526 	aNumberOfUnconvertibleCharacters = 0;
  1527 	TUint aOutputConversionFlags = CCnvCharacterSetConverter::KStateDefault;
  1528 	aIndexOfFirstByteOfFirstUnconvertibleCharacter=-1;
  1529 	TUint internalInputConversionFlags = 0;
  1530 	TInt numberOfForeignBytesConsumed=0;
  1531 	TPtrC8 remainderOfForeign(aForeign);
  1532 	TInt returnValue;
  1533 	TBool flag = EFalse;
  1534 	TBool isSkipMatchSequence = EFalse;
  1535 	const SCnvConversionData* convData;
  1536 	//Set the iscii conversion data and escape sequence for Devanagari.
  1537 	TFixedArray<CnvUtilities::SState,KNumberOfIndicCharactersetsSupported> modals;
  1538 	modals[0].iConversionData = &conversionData;
  1539 	modals[0].iEscapeSequence = &KEscapeSequenceDevanagari();
  1540 
  1541 	aUnicode.SetLength(0);
  1542 	
  1543 	//The internal input conversion flag for conversion is set to CCnvCharacterSetConverter::EInputConversionFlagAppend
  1544 	//so that for each conversion in the conversion loop, the generated conversion buffer is appened to the aUnicode buffer.
  1545 	internalInputConversionFlags |= CCnvCharacterSetConverter::EInputConversionFlagAppend;
  1546 	if (aForeign.Length()==0)
  1547 	{
  1548 		return 0;
  1549 	}
  1550 	//Get the conversion data from the previous call else the conversion data is set to the default 
  1551 	//conversion data, i.e. Devanagari.
  1552 	convData=(aState!=CCnvCharacterSetConverter::KStateDefault)? REINTERPRET_CAST(const SCnvConversionData*, aState): modals[0].iConversionData;
  1553 	FOREVER
  1554 	{
  1555 		TBuf8<KMaximumLengthOfIntermediateBuffer> intermediateBuffer;
  1556 		TUint numberOfForeignBytesConsumedThisTime = 0;
  1557 		if((remainderOfForeign.Length() >=KMaximumLengthOfIntermediateBuffer) && (aUnicode.MaxLength()-aUnicode.Length() >=KMaximumLengthOfIntermediateBuffer))
  1558 		{
  1559 			numberOfForeignBytesConsumedThisTime = KMaximumLengthOfIntermediateBuffer;
  1560 			intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
  1561 			//If the intermediate buffer is a part of truncated buffer sequence but the
  1562 			//actual input buffer is not truncated then truncated sequence is not converted.
  1563 			//The intermediate buffer is modified so as not to contain the truncated sequence.
  1564 			
  1565 			flag = IsTruncatedDoubleByteIsciiSequence(intermediateBuffer);
  1566 			if(flag)
  1567 				{
  1568 					numberOfForeignBytesConsumedThisTime --;
  1569 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);					
  1570 				}
  1571 			
  1572 		}
  1573 		else
  1574 		{
  1575 			flag = IsTruncatedDoubleByteIsciiSequence(remainderOfForeign.Left(aUnicode.MaxLength()-aUnicode.Length()));
  1576 			if(!flag)
  1577 				{
  1578 				numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length();
  1579 				intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
  1580 				}
  1581 			else
  1582 				{
  1583 				if(aForeign.Length()>(numberOfForeignBytesConsumed+aUnicode.Length()))
  1584 					{
  1585 					numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length()-1;
  1586 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
  1587 					break;
  1588 					}
  1589 				else
  1590 					{
  1591 					numberOfForeignBytesConsumedThisTime = aUnicode.MaxLength()-aUnicode.Length();
  1592 					intermediateBuffer = remainderOfForeign.Left(numberOfForeignBytesConsumedThisTime);
  1593 					}
  1594 				}
  1595 		}			
  1596 		
  1597 		//The input intermediate iscii buffer is modified with the search and replace
  1598 		//buffers. It is required for supporting multibyte iscii sequences.		
  1599 		FindAndModifyBuffer(intermediateBuffer);
  1600 		TPtrC8 remainderOfForeignInternal(intermediateBuffer);
  1601 		TPtrC8 homogeneousRun;
  1602 		const TInt startOfNextEscapeSequence=intermediateBuffer.Locate(KControlCharacterEscape);
  1603 		if (startOfNextEscapeSequence!=0) 
  1604 			{
  1605 			if (startOfNextEscapeSequence==KErrNotFound)
  1606 				{
  1607 				homogeneousRun.Set(remainderOfForeignInternal);
  1608 				remainderOfForeignInternal.Set(NULL, 0);
  1609 				}
  1610 			else
  1611 				{
  1612 				__ASSERT_DEBUG(startOfNextEscapeSequence>0, User::Panic(KPanicReason,EPanicBadStartOfNextEscapeSequence));
  1613 				homogeneousRun.Set(remainderOfForeignInternal.Left(startOfNextEscapeSequence));
  1614 				remainderOfForeignInternal.Set(remainderOfForeignInternal.Mid(startOfNextEscapeSequence));
  1615 				}
  1616 			isSkipMatchSequence = ETrue;
  1617 			}
  1618 		FOREVER
  1619 			{
  1620 			if(!isSkipMatchSequence)
  1621 				{
  1622 				if (!NextHomogeneousForeignRun(convData, numberOfForeignBytesConsumed, homogeneousRun, 
  1623 											remainderOfForeignInternal, modals.Array(), aOutputConversionFlags))
  1624 					{
  1625 					break;
  1626 					}
  1627 				}
  1628 			HandleHomogeneousRun( convData, aDefaultEndiannessOfForeignCharacters, aUnicode, homogeneousRun, aNumberOfUnconvertibleCharacters, 
  1629 								aIndexOfFirstByteOfFirstUnconvertibleCharacter,aOutputConversionFlags,internalInputConversionFlags,
  1630 								numberOfForeignBytesConsumed,returnValue);
  1631 			if(returnValue<0)
  1632 				{
  1633 				return returnValue;
  1634 				}
  1635 			isSkipMatchSequence = EFalse;
  1636 			}
  1637 		if(returnValue > 0)
  1638 			break;
  1639 		if ((!flag && (numberOfForeignBytesConsumedThisTime != KMaximumLengthOfIntermediateBuffer)) || (flag && (numberOfForeignBytesConsumedThisTime != (KMaximumLengthOfIntermediateBuffer-1) )))
  1640 			break;
  1641 		remainderOfForeign.Set(aForeign.Mid(numberOfForeignBytesConsumed));				
  1642 	}
  1643 	//If the number of iscii bytes consumed by the conversion is zero also the output conversion
  1644 	//flag is not set to EOutputConversionFlagInputIsTruncated, then return EErrorIllFormedInput.
  1645 	if ((numberOfForeignBytesConsumed==0) && (aOutputConversionFlags&CCnvCharacterSetConverter::EOutputConversionFlagInputIsTruncated))
  1646 		{
  1647 		return CCnvCharacterSetConverter::EErrorIllFormedInput;
  1648 		}
  1649 	//Set the conversion data sothat next time when ConvertToUnicode() is called,
  1650 	//will use the previous conversion data.
  1651 	aState=REINTERPRET_CAST(TInt, convData);
  1652 	return aForeign.Length()-numberOfForeignBytesConsumed;
  1653     }
  1654 
  1655 // -----------------------------------------------------------------------------
  1656 // IsInThisCharacterSetL()
  1657 //The method tells how probable it is that a sample piece of text is encoded in this character set.
  1658 //On return aConfidenceLevel, indicates how confident the function is about its return value. For
  1659 //iscii it is the default implementation and it does not implement the autodetect functionality.
  1660 //Loaded and called by the character conversion framework.
  1661 //
  1662 // Returns: EFalse: To tell CCnvCharacterSetConverter::AutoDetectCharacterSetL()  that the plug-in DLL
  1663 //                          is not implementing a function of this signature and is therefore empty.
  1664 //
  1665 // -----------------------------------------------------------------------------
  1666 //
  1667 
  1668 
  1669 //Default implementation for IsInThisCharacterSetL()
  1670 
  1671 EXPORT_C TBool IsInThisCharacterSetL(
  1672          TBool& aSetToTrue, 
  1673          TInt& aConfidenceLevel, 
  1674          const TDesC8& )
  1675 	{
  1676     	aSetToTrue = EFalse;
  1677 	aConfidenceLevel = 0;
  1678 	return EFalse;
  1679 	}
  1680 
  1681 EXPORT_C void Reserved_2()
  1682 	{
  1683 	}
  1684 
  1685 EXPORT_C void Reserved_3()
  1686 	{
  1687 	}
  1688 
  1689 EXPORT_C void Reserved_4()
  1690 	{
  1691 	}
  1692 
  1693 EXPORT_C void Reserved_5()
  1694 	{
  1695 	}
  1696 
  1697 EXPORT_C void Reserved_6()
  1698 	{
  1699 	}
  1700 
  1701 EXPORT_C void Reserved_7()
  1702 	{
  1703 	}
  1704 
  1705 EXPORT_C void Reserved_8()
  1706 	{
  1707 	}
  1708 
  1709 #endif //EKA2