os/textandloc/charconvfw/fatfilenameconversionplugins/src/cp54936_unicodeconv.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.
sl@0
     1
/*
sl@0
     2
* Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     3
* All rights reserved.
sl@0
     4
* This component and the accompanying materials are made available
sl@0
     5
* under the terms of "Eclipse Public License v1.0"
sl@0
     6
* which accompanies this distribution, and is available
sl@0
     7
* at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     8
*
sl@0
     9
* Initial Contributors:
sl@0
    10
* Nokia Corporation - initial contribution.
sl@0
    11
*
sl@0
    12
* Contributors:
sl@0
    13
*
sl@0
    14
* Description: 
sl@0
    15
*
sl@0
    16
*/
sl@0
    17
// There are 2 reasons why not use existing unicodeconv.cpp:
sl@0
    18
// 1) "unicode->foreign" in existing unicodeconv.cpp is quite slow, especially
sl@0
    19
//    for huge code pages (e.g, Asia code pages). See INC127598.
sl@0
    20
//
sl@0
    21
// 2) GB18030 has 32-bit code that existing unicodeconv.cpp cannot handle.
sl@0
    22
//
sl@0
    23
// The algorithm of this special version unicodeconv.cpp is straightforward:
sl@0
    24
// 1) foreign->unicode:
sl@0
    25
//    1.1) 1 byte/2 byte->unicode bmp: use existing mechanism; mapping table in
sl@0
    26
//              "cp54936_2byte_tounicode.cpp", which is generated with command
sl@0
    27
//              "perl -w ..\group\FatConversionTable.pl cp54936_2byte.txt".
sl@0
    28
//
sl@0
    29
//    1.2) 4 byte->unicode bmp: convert the 4-byte code to a 16-bit index, then
sl@0
    30
//              search into the mapping table in "cp54936_4byte_tounicode.cpp",
sl@0
    31
//              which is generated with command
sl@0
    32
//              "perl -w ..\group\cp54936_4byte_tounicode.pl cp54936_4byte.txt".
sl@0
    33
//
sl@0
    34
//    1.3) 4 byte->unicode non-bmp: calculate with formula in this file.
sl@0
    35
//
sl@0
    36
// 2) unicode->foreign:
sl@0
    37
//    2.1) unicode bmp->1/2/4 byte: the huge table in "cp54936_allbmp_fromunicode.cpp"
sl@0
    38
//              can map directly, which is generated with command
sl@0
    39
//              "perl -w ..\group\cp54936_allbmp_fromunicode.pl cp54936_2byte.txt cp54936_4byte.txt".
sl@0
    40
//
sl@0
    41
//    2.2) unicode non-bmp->4 byte: calculate with formula in this file.
sl@0
    42
//
sl@0
    43
// The function cp54936_2byte_tounicode.cpp::TConvDataStruct::
sl@0
    44
// ConvertSingleUnicode() is not used anymore. It's reserved just because not
sl@0
    45
// changing the tool FatConversionTable.pl.
sl@0
    46
//
sl@0
    47
// About the mapping table "cp54936_2byte.txt" and "cp54936_4byte.txt":
sl@0
    48
// 1) All Private Used Area (PUA) code points are reserved.
sl@0
    49
// 2) All GB18030 code points that mapping to undefined Unicode are reserved.
sl@0
    50
//
sl@0
    51
//
sl@0
    52
// About the formula for non-bmp calculation:
sl@0
    53
// 1) All code points from 0x10000 to 0x10FFFF are supported.
sl@0
    54
// 2) Code points in 0x10000-0x1FFFF and 0x30000-0x10FFFF are summarized from
sl@0
    55
//    the GB18030 standard, since the standard does not define the mapping for
sl@0
    56
//    code points out of 0x20000-0x2FFFF.
sl@0
    57
sl@0
    58
sl@0
    59
#include <e32std.h>
sl@0
    60
#include <e32def.h>
sl@0
    61
#include <e32des8.h> 
sl@0
    62
#include "unicodeconv.h"
sl@0
    63
#include "cp54936.h"
sl@0
    64
sl@0
    65
sl@0
    66
enum TFccPanic
sl@0
    67
	{
sl@0
    68
	EBadForeignCode = 0,
sl@0
    69
	E4ByteIndexOutOfRange,
sl@0
    70
	EPanicBadIndices1,
sl@0
    71
	EInavlidUnicodeValue
sl@0
    72
	};
sl@0
    73
void Panic(TFccPanic aPanic)
sl@0
    74
	{
sl@0
    75
sl@0
    76
	User::Panic(_L("FatCharsetConv"),aPanic);
sl@0
    77
	}
sl@0
    78
sl@0
    79
sl@0
    80
//replacement character to be used when unicode cannot be converted
sl@0
    81
const TUint8 KForeignReplacement = 0x5F;
sl@0
    82
sl@0
    83
const TUint8 KU10000Byte1 = 0x90;
sl@0
    84
const TUint8 KU10000Byte2 = 0x30;
sl@0
    85
const TUint8 KU10000Byte3 = 0x81;
sl@0
    86
const TUint8 KU10000Byte4 = 0x30;
sl@0
    87
sl@0
    88
inline TBool IsSupplementary(TUint aChar)
sl@0
    89
/**
sl@0
    90
@param aChar The 32-bit code point value of a Unicode character.
sl@0
    91
sl@0
    92
@return True, if aChar is supplementary character; false, otherwise.
sl@0
    93
*/
sl@0
    94
	{
sl@0
    95
	return (aChar > 0xFFFF);
sl@0
    96
	}
sl@0
    97
sl@0
    98
inline TBool IsSurrogate(TText16 aInt16)
sl@0
    99
/**
sl@0
   100
@return True, if aText16 is high surrogate or low surrogate; false, otherwise.
sl@0
   101
*/
sl@0
   102
	{
sl@0
   103
	return (aInt16 & 0xF800) == 0xD800;
sl@0
   104
	}
sl@0
   105
sl@0
   106
inline TBool IsHighSurrogate(TText16 aInt16)
sl@0
   107
/**
sl@0
   108
@return True, if aText16 is high surrogate; false, otherwise.
sl@0
   109
*/
sl@0
   110
	{
sl@0
   111
	return (aInt16 & 0xFC00) == 0xD800;
sl@0
   112
	}
sl@0
   113
sl@0
   114
inline TBool IsLowSurrogate(TText16 aInt16)
sl@0
   115
/**
sl@0
   116
@return True, if aText16 is low surrogate; false, otherwise.
sl@0
   117
*/
sl@0
   118
	{
sl@0
   119
	return (aInt16 & 0xFC00) == 0xDC00;
sl@0
   120
	}
sl@0
   121
sl@0
   122
inline TUint JoinSurrogate(TText16 aHighSurrogate, TText16 aLowSurrogate)
sl@0
   123
/**
sl@0
   124
Combine a high surrogate and a low surrogate into a supplementary character.
sl@0
   125
sl@0
   126
@return The 32-bit code point value of the generated Unicode supplementary
sl@0
   127
        character.
sl@0
   128
*/
sl@0
   129
	{
sl@0
   130
	return ((aHighSurrogate - 0xD7F7) << 10) + aLowSurrogate;
sl@0
   131
	}
sl@0
   132
sl@0
   133
inline TText16 GetHighSurrogate(TUint aChar)
sl@0
   134
/**
sl@0
   135
Retrieve the high surrogate of a supplementary character.
sl@0
   136
sl@0
   137
@param aChar The 32-bit code point value of a Unicode character.
sl@0
   138
sl@0
   139
@return High surrogate of aChar, if aChar is a supplementary character; 
sl@0
   140
        aChar itself, if aChar is not a supplementary character.
sl@0
   141
*/
sl@0
   142
	{
sl@0
   143
	return STATIC_CAST(TText16, 0xD7C0 + (aChar >> 10));
sl@0
   144
	}
sl@0
   145
sl@0
   146
inline TText16 GetLowSurrogate(TUint aChar)
sl@0
   147
/**
sl@0
   148
Retrieve the low surrogate of a supplementary character.
sl@0
   149
sl@0
   150
@param aChar The 32-bit code point value of a Unicode character.
sl@0
   151
sl@0
   152
@return Low surrogate of aChar, if aChar is a supplementary character; 
sl@0
   153
        zero, if aChar is not a supplementary character.
sl@0
   154
*/
sl@0
   155
	{
sl@0
   156
	return STATIC_CAST(TText16, 0xDC00 | (aChar & 0x3FF));
sl@0
   157
	}
sl@0
   158
sl@0
   159
//This function converts from Unicoded characters, to foreign characters and adds them into a descriptor
sl@0
   160
EXPORT_C void UnicodeConv::ConvertFromUnicodeL(TDes8& aForeign, const TDesC16& aUnicode)
sl@0
   161
	{
sl@0
   162
    UnicodeConv::ConvertFromUnicodeL(aForeign, aUnicode, ETrue);
sl@0
   163
    }
sl@0
   164
sl@0
   165
//This function converts from Unicoded characters, to foreign characters and adds them into a descriptor
sl@0
   166
EXPORT_C TInt UnicodeConv::ConvertFromUnicodeL(TDes8& aForeign, const TDesC16& aUnicode, TBool leaveWhenOverflow)
sl@0
   167
	{
sl@0
   168
	const TInt length = aUnicode.Length();
sl@0
   169
	const TUint16* unicode = aUnicode.Ptr();
sl@0
   170
	const TUint16* guard = unicode + length;
sl@0
   171
	
sl@0
   172
	TUint8* foreign = const_cast<TUint8*>(aForeign.Ptr());
sl@0
   173
	TUint8* foreignguard = foreign + aForeign.MaxLength();
sl@0
   174
	
sl@0
   175
	//loop going through the character of the unicode descriptor
sl@0
   176
	while (unicode < guard)
sl@0
   177
		{
sl@0
   178
		TUint32 unicodeChar = *unicode++;
sl@0
   179
		if (IsHighSurrogate(unicodeChar))
sl@0
   180
			{
sl@0
   181
			if (unicode >= guard || !IsLowSurrogate(*unicode))
sl@0
   182
				{
sl@0
   183
				if (foreign >= foreignguard)
sl@0
   184
					{
sl@0
   185
                    aForeign.SetLength(foreign-aForeign.Ptr());
sl@0
   186
					if (leaveWhenOverflow)
sl@0
   187
						User::Leave(KErrOverflow);
sl@0
   188
                    else
sl@0
   189
                    	return KErrOverflow;
sl@0
   190
					}
sl@0
   191
				*foreign++ = KForeignReplacement;
sl@0
   192
				continue;
sl@0
   193
				}
sl@0
   194
			unicodeChar = JoinSurrogate(unicodeChar, *unicode++);
sl@0
   195
			}
sl@0
   196
		if (IsLowSurrogate(unicodeChar))
sl@0
   197
			{
sl@0
   198
			if (foreign >= foreignguard)
sl@0
   199
				{
sl@0
   200
				aForeign.SetLength(foreign-aForeign.Ptr());
sl@0
   201
				if (leaveWhenOverflow)
sl@0
   202
					User::Leave(KErrOverflow);
sl@0
   203
				else
sl@0
   204
					return KErrOverflow;
sl@0
   205
				}
sl@0
   206
			*foreign++ = KForeignReplacement;
sl@0
   207
			continue;
sl@0
   208
			}
sl@0
   209
		
sl@0
   210
		TUint8 b1, b2, b3, b4;		// byte 1,2,3,4 of result GB18030 code.
sl@0
   211
		TInt count;					// byte count of result GB18030 code; can be 1, 2 or 4.
sl@0
   212
		
sl@0
   213
		// unicode to cp54936
sl@0
   214
		if (IsSupplementary(unicodeChar))
sl@0
   215
			{
sl@0
   216
			unicodeChar -= 0x10000;
sl@0
   217
			b4 = unicodeChar % 10 + KU10000Byte4;
sl@0
   218
			unicodeChar /= 10;
sl@0
   219
			b3 = unicodeChar % 126 + KU10000Byte3;
sl@0
   220
			unicodeChar /= 126;
sl@0
   221
			b2 = unicodeChar % 10 + KU10000Byte2;
sl@0
   222
			b1 = unicodeChar / 10 + KU10000Byte1;
sl@0
   223
			count = 4;
sl@0
   224
			}
sl@0
   225
		else
sl@0
   226
			{
sl@0
   227
			TUint32 foreignChar;
sl@0
   228
			foreignChar = KMappingTableUnicodeBmp2CP54936[unicodeChar];
sl@0
   229
			b1 = ((foreignChar >> 24) & 0xFF);
sl@0
   230
			b2 = ((foreignChar >> 16) & 0xFF);
sl@0
   231
			b3 = ((foreignChar >> 8) & 0xFF);
sl@0
   232
			b4 = (foreignChar & 0xFF);
sl@0
   233
			count = 1;
sl@0
   234
			if (b1)
sl@0
   235
				{
sl@0
   236
				count = 4;
sl@0
   237
				}
sl@0
   238
			else
sl@0
   239
				{
sl@0
   240
				__ASSERT_DEBUG(b2==0, Panic(EBadForeignCode));
sl@0
   241
				if (b3)
sl@0
   242
					{
sl@0
   243
					count = 2;
sl@0
   244
					}
sl@0
   245
				}
sl@0
   246
			}
sl@0
   247
		
sl@0
   248
		if (foreign + count > foreignguard)
sl@0
   249
			{
sl@0
   250
			aForeign.SetLength(foreign-aForeign.Ptr());
sl@0
   251
            if (leaveWhenOverflow)
sl@0
   252
            	User::Leave(KErrOverflow);
sl@0
   253
            else
sl@0
   254
            	return KErrOverflow;
sl@0
   255
			}
sl@0
   256
		if (count == 4)
sl@0
   257
			{
sl@0
   258
			*foreign++ = b1;
sl@0
   259
			*foreign++ = b2;
sl@0
   260
			}
sl@0
   261
		if (count >= 2)
sl@0
   262
			*foreign++ = b3;
sl@0
   263
		*foreign++ = b4;
sl@0
   264
		}
sl@0
   265
	aForeign.SetLength(foreign-aForeign.Ptr());
sl@0
   266
	return KErrNone;
sl@0
   267
	}
sl@0
   268
sl@0
   269
sl@0
   270
//This function converts from foreign characters into unicode and adds them into a descriptor
sl@0
   271
EXPORT_C void UnicodeConv::ConvertToUnicodeL(TDes16& aUnicode, const TDesC8& aForeign)
sl@0
   272
	{
sl@0
   273
    UnicodeConv::ConvertToUnicodeL(aUnicode, aForeign, ETrue);
sl@0
   274
    }
sl@0
   275
sl@0
   276
//This function converts from foreign characters into unicode and adds them into a descriptor
sl@0
   277
EXPORT_C TInt UnicodeConv::ConvertToUnicodeL(TDes16& aUnicode, const TDesC8& aForeign, TBool leaveWhenOverflow)
sl@0
   278
	{
sl@0
   279
	const TInt foreignLength = aForeign.Length();
sl@0
   280
	const TUint8* foreign = aForeign.Ptr();
sl@0
   281
	const TUint8* guard = foreign + foreignLength;
sl@0
   282
	
sl@0
   283
	TUint16* unicode = const_cast<TUint16*>(aUnicode.Ptr());
sl@0
   284
	TUint16* unicodeguard = unicode + aUnicode.MaxLength();
sl@0
   285
	
sl@0
   286
	TUint8 b1, b2, b3, b4;
sl@0
   287
	enum TCodeType
sl@0
   288
	{
sl@0
   289
	E1Byte = 0,
sl@0
   290
	E2Byte,
sl@0
   291
	E4ByteBmp,
sl@0
   292
	E4ByteSupplementary,
sl@0
   293
	EError,
sl@0
   294
	};
sl@0
   295
	TCodeType codetype;
sl@0
   296
	TUint32 unicodeChar;
sl@0
   297
sl@0
   298
	//loop going through the characters of the foreign descriptor
sl@0
   299
	while (foreign < guard)
sl@0
   300
		{
sl@0
   301
		// roughly, detect which area the foreign code belongs to
sl@0
   302
		b1 = *foreign++;
sl@0
   303
		if (b1 <= 0x7F)
sl@0
   304
			codetype = E1Byte;
sl@0
   305
		else if (b1 == 0x80 || b1 > 0xFE)
sl@0
   306
			codetype = EError;
sl@0
   307
		else if (foreign >= guard)
sl@0
   308
			codetype = EError;
sl@0
   309
		else
sl@0
   310
			{
sl@0
   311
			b2 = *foreign++;
sl@0
   312
			if (b2 >= 0x40 && b2 <= 0xFE && b2 != 0x7F)
sl@0
   313
				codetype = E2Byte;
sl@0
   314
			else if (b2 < 0x30 || b2 > 0x39)
sl@0
   315
				codetype = EError;
sl@0
   316
			else if (foreign+1 >= guard)
sl@0
   317
				codetype = EError;
sl@0
   318
			else
sl@0
   319
				{
sl@0
   320
				b3 = *foreign++;
sl@0
   321
				if (b3 < 0x81 || b3 > 0xFE)
sl@0
   322
					codetype = EError;
sl@0
   323
				else
sl@0
   324
					{
sl@0
   325
					b4 = *foreign++;
sl@0
   326
					if (b4 < 0x30 || b4 > 0x39)
sl@0
   327
						codetype = EError;
sl@0
   328
					else if (b1 >= 0x81 && b1 <= 0x84)		// 0x81308130-0x8439FE39
sl@0
   329
						codetype = E4ByteBmp;
sl@0
   330
					else if (b1 >= 0x90 && b1 <= 0xE3)		// 0x90308130-0xE339FE39
sl@0
   331
						codetype = E4ByteSupplementary;
sl@0
   332
					else
sl@0
   333
						codetype = EError;					// others are reserved
sl@0
   334
					}
sl@0
   335
				}
sl@0
   336
			}
sl@0
   337
		
sl@0
   338
		// cp54936 to unicode
sl@0
   339
		if (codetype == E1Byte)
sl@0
   340
			{
sl@0
   341
			unicodeChar = b1;
sl@0
   342
			}
sl@0
   343
		else if (codetype == E2Byte)
sl@0
   344
			{
sl@0
   345
			// conventional algorithm used in FatCharsetConv
sl@0
   346
			const TLeadOrSingle* structPtr = TConvDataStruct::KFirstByteConversions + (b1-0x80);
sl@0
   347
			if (structPtr->iUnicodeIfSingle)
sl@0
   348
				unicodeChar = structPtr->iUnicodeIfSingle;
sl@0
   349
			else if (TConvDataStruct::KMinTrailByte <= b2 && b2 <= TConvDataStruct::KMaxTrailByte)
sl@0
   350
				unicodeChar = TConvDataStruct::KDoubleByteConversions[structPtr->iDoubleByteIndex + (b2 - TConvDataStruct::KMinTrailByte)];
sl@0
   351
			else
sl@0
   352
				unicodeChar = 0xFFFD;
sl@0
   353
			}
sl@0
   354
		else if (codetype == E4ByteBmp)
sl@0
   355
			{
sl@0
   356
			TUint index = (b1-0x81)*12600 + (b2-0x30)*1260 + (b3-0x81)*10 + (b4-0x30);
sl@0
   357
			__ASSERT_DEBUG(index<39420, Panic(E4ByteIndexOutOfRange));
sl@0
   358
			unicodeChar = KMappingTable4ByteBmp2Unicode[index];
sl@0
   359
			}
sl@0
   360
		else if (codetype == E4ByteSupplementary)
sl@0
   361
			{
sl@0
   362
			unicodeChar = 0x10000 + (b1 - KU10000Byte1) * 12600 +
sl@0
   363
									(b2 - KU10000Byte2) * 1260 +
sl@0
   364
									(b3 - KU10000Byte3) * 10 +
sl@0
   365
									(b4 - KU10000Byte4);
sl@0
   366
			__ASSERT_DEBUG(unicodeChar >= 0x10000 && unicodeChar <= 0x10FFFF, Panic(EInavlidUnicodeValue));
sl@0
   367
			}
sl@0
   368
		else
sl@0
   369
			{
sl@0
   370
			unicodeChar = 0xFFFD;
sl@0
   371
			}
sl@0
   372
		
sl@0
   373
		// append to output buffer
sl@0
   374
		if (IsSupplementary(unicodeChar))
sl@0
   375
			{
sl@0
   376
			if (unicode + 1 >= unicodeguard)
sl@0
   377
				{
sl@0
   378
				aUnicode.SetLength(unicode-aUnicode.Ptr());
sl@0
   379
				if (leaveWhenOverflow)
sl@0
   380
					User::Leave(KErrOverflow);
sl@0
   381
				else
sl@0
   382
					return KErrOverflow;
sl@0
   383
				}
sl@0
   384
			*unicode++ = GetHighSurrogate(unicodeChar);
sl@0
   385
			*unicode++ = GetLowSurrogate(unicodeChar);
sl@0
   386
			}
sl@0
   387
		else
sl@0
   388
			{
sl@0
   389
			if (unicode >= unicodeguard)
sl@0
   390
				{
sl@0
   391
				aUnicode.SetLength(unicode-aUnicode.Ptr());
sl@0
   392
                if (leaveWhenOverflow)
sl@0
   393
                	User::Leave(KErrOverflow);
sl@0
   394
                else
sl@0
   395
                	return KErrOverflow;
sl@0
   396
				}
sl@0
   397
			*unicode++ = unicodeChar;
sl@0
   398
			}
sl@0
   399
		}
sl@0
   400
	aUnicode.SetLength(unicode-aUnicode.Ptr());
sl@0
   401
	return KErrNone;
sl@0
   402
	}
sl@0
   403
sl@0
   404
EXPORT_C TBool UnicodeConv::IsLegalShortNameCharacter (TUint aCharacter)
sl@0
   405
	{
sl@0
   406
	//1. aCharacter >= 0x0080 
sl@0
   407
	if (aCharacter>=0x0080)
sl@0
   408
		{
sl@0
   409
		// Since all Unicode characters can be mapped to GB18030, so no need to
sl@0
   410
		// test the converting.
sl@0
   411
		if (aCharacter <= 0x10FFFF && !IsSurrogate(aCharacter))
sl@0
   412
			return ETrue;
sl@0
   413
		else
sl@0
   414
			return EFalse;
sl@0
   415
		}
sl@0
   416
sl@0
   417
    // For most common cases: 
sl@0
   418
    // Note: lower case characters are considered legal DOS char here. 
sl@0
   419
	if ((aCharacter>='a' && aCharacter<='z') || 
sl@0
   420
	    (aCharacter>='A' && aCharacter<='Z') || 
sl@0
   421
	    (aCharacter>='0' && aCharacter<='9'))
sl@0
   422
			{
sl@0
   423
			return ETrue;
sl@0
   424
			}
sl@0
   425
    // Checking for illegal chars: 
sl@0
   426
    // 2. aCharacter <= 0x20 
sl@0
   427
    // Note: leading 0x05 byte should be guarded by callers of this function 
sl@0
   428
    //  as the information of the position of the character is required. 
sl@0
   429
	if (aCharacter < 0x20)
sl@0
   430
		return EFalse;
sl@0
   431
	// Space (' ') is not considered as a legal DOS char here.
sl@0
   432
	if (aCharacter == 0x20)
sl@0
   433
		return EFalse;
sl@0
   434
	
sl@0
   435
	// 3. 0x20 < aCharacter < 0x80 
sl@0
   436
    // According to FAT Spec, "following characters are not legal in any bytes of DIR_Name": 
sl@0
   437
    switch (aCharacter) 
sl@0
   438
            { 
sl@0
   439
            case 0x22:        // '"' 
sl@0
   440
            case 0x2A:        // '*' 
sl@0
   441
            case 0x2B:        // '+' 
sl@0
   442
            case 0x2C:        // ',' 
sl@0
   443
            //case 0x2E:        // '.'   // Although '.' is not allowed in any bytes of DIR_Name, it 
sl@0
   444
                                         // is a valid character in short file names. 
sl@0
   445
            case 0x2F:        // '/' 
sl@0
   446
            case 0x3A:        // ':' 
sl@0
   447
            case 0x3B:        // ';' 
sl@0
   448
            case 0x3C:        // '<' 
sl@0
   449
            case 0x3D:        // '=' 
sl@0
   450
            case 0x3E:        // '>' 
sl@0
   451
            case 0x3F:        // '?' 
sl@0
   452
            case 0x5B:        // '[' 
sl@0
   453
            case 0x5C:        // '\' 
sl@0
   454
            case 0x5D:        // ']' 
sl@0
   455
            case 0x7C:        // '|' 
sl@0
   456
            	return EFalse; 
sl@0
   457
            default: 
sl@0
   458
            	return ETrue; 
sl@0
   459
            } 
sl@0
   460
	}		
sl@0
   461