os/ossrv/lowlevellibsandfws/apputils/src/StringPoolImplementation.h
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2001-2009 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#ifndef __STRINGPOOLIMPLEMENTATION_H__
sl@0
    17
#define __STRINGPOOLIMPLEMENTATION_H__
sl@0
    18
sl@0
    19
#include <e32base.h>
sl@0
    20
#include <stringpool.h>
sl@0
    21
#include <stringtablesupport.h>
sl@0
    22
sl@0
    23
/**
sl@0
    24
This is an 8 bit value with every bit set except bit 5 (value
sl@0
    25
32). The ASCII codes of upper and lower case characters differ by
sl@0
    26
32.
sl@0
    27
@internalComponent 
sl@0
    28
*/
sl@0
    29
const TUint KCaseInsensitive = 223;
sl@0
    30
sl@0
    31
/**
sl@0
    32
@internalComponent 
sl@0
    33
*/
sl@0
    34
const TInt KHashModulo = 32;
sl@0
    35
sl@0
    36
/**
sl@0
    37
Bit 0 represents if it is a table (set) or not (unset).
sl@0
    38
Bit 1 represents case sensitivity (set) or not (unset)
sl@0
    39
sl@0
    40
For non-static table values (dynamic strings), the remaining bits 
sl@0
    41
are interpreted as a pointer to a CStringPoolNode (which is assumed 
sl@0
    42
to be word aligned). 
sl@0
    43
sl@0
    44
Bits 2-20 represent the table index.
sl@0
    45
Bits 21-31 represent the table uid (that is the address of the table).
sl@0
    46
*/
sl@0
    47
sl@0
    48
/**
sl@0
    49
Bit 1 is used in tokens to indicate case sensitivity. So mask it
sl@0
    50
out to get the node.
sl@0
    51
@internalComponent 
sl@0
    52
*/
sl@0
    53
const TUint KTokenToNode = 0xfffffffd;
sl@0
    54
sl@0
    55
/**
sl@0
    56
Determine if this id corresponds to a table.
sl@0
    57
@internalComponent
sl@0
    58
*/
sl@0
    59
#define IS_TABLE_ENTRY(tableId)(tableId & 0x01)
sl@0
    60
sl@0
    61
/**
sl@0
    62
Obtain the Table Index.
sl@0
    63
@internalComponent
sl@0
    64
*/
sl@0
    65
#define TABLE_INDEX(val)((val & 0xffffc)>>2)
sl@0
    66
sl@0
    67
/**
sl@0
    68
Obtain the Table UID.
sl@0
    69
@internalComponent
sl@0
    70
*/
sl@0
    71
#define TABLE_UID(val)(val>>20)
sl@0
    72
sl@0
    73
/**
sl@0
    74
This is used to mark the reference count of a node so that is will not be deleted when closed
sl@0
    75
@internalComponent
sl@0
    76
*/
sl@0
    77
const TUint KMarkedForNoDeleted = 0xffff;
sl@0
    78
sl@0
    79
sl@0
    80
sl@0
    81
class RStringTokenEither;
sl@0
    82
sl@0
    83
/**
sl@0
    84
@internalComponent
sl@0
    85
*/
sl@0
    86
struct TStringIdMap
sl@0
    87
	{
sl@0
    88
	TInt32 iSourceTableVal;
sl@0
    89
	TInt32 iTargetTableVal;
sl@0
    90
	};
sl@0
    91
sl@0
    92
sl@0
    93
/**
sl@0
    94
Internal node class
sl@0
    95
@internalComponent
sl@0
    96
*/
sl@0
    97
NONSHARABLE_CLASS(CStringPoolNode) : public CBase
sl@0
    98
	{
sl@0
    99
public:
sl@0
   100
	~CStringPoolNode();
sl@0
   101
sl@0
   102
 public:
sl@0
   103
	HBufC8* iDes;
sl@0
   104
	TUint16 iRefcount;
sl@0
   105
	TUint8  iHash;
sl@0
   106
	};
sl@0
   107
sl@0
   108
/**
sl@0
   109
@internalComponent
sl@0
   110
*/
sl@0
   111
NONSHARABLE_CLASS(CStringPoolImplementation) : public CBase
sl@0
   112
	{
sl@0
   113
public:
sl@0
   114
	CStringPoolImplementation();
sl@0
   115
	// Destructor
sl@0
   116
	~CStringPoolImplementation();
sl@0
   117
sl@0
   118
	// Constructs a string pool
sl@0
   119
	static CStringPoolImplementation* NewL();
sl@0
   120
sl@0
   121
	// USed for cleaning up when a OpenTableL leaves
sl@0
   122
	static void CleanupHashCS(TAny* aImplementation);
sl@0
   123
	static void CleanupHashCI(TAny* aImplementation);
sl@0
   124
	static void CleanupIdMap(TAny* aImplementation);
sl@0
   125
	static void CleanUpHash(RPointerArray <RStringTokenEither>* aHashCleanup, CArrayFixSeg<RStringTokenEither>* hash[KHashModulo]);
sl@0
   126
sl@0
   127
	// Adds a table to the pool. (Note this currently only works once)
sl@0
   128
	void AddTableL(const TStringTable& aTable);
sl@0
   129
	void AddCallBackL( MStringPoolCloseCallBack& aCallBack);
sl@0
   130
sl@0
   131
sl@0
   132
	// Find FirstVal given duplicate val
sl@0
   133
	TInt32 FindFirstValFromDuplicate(TInt32 aDuplicateVal) const;
sl@0
   134
sl@0
   135
	// Find table index Val given first val & table UID
sl@0
   136
	TInt FindTableIndexFromFirstVal(TInt32 aDuplicateVal, TInt aTableUid) const;
sl@0
   137
sl@0
   138
sl@0
   139
	TInt16 TableUid(const TStringTable& aTable) const;
sl@0
   140
sl@0
   141
	const TStringTable& TableRef(TInt32 aVal) const;
sl@0
   142
sl@0
   143
	// Looks up a particular index in the pre-loaded tables
sl@0
   144
	const TDesC8& TableLookup(TInt aIndex, TInt aTableUid) const;
sl@0
   145
sl@0
   146
	// Finds or creates a string. Increments the reference count if needed.
sl@0
   147
	RStringTokenEither OpenL( const TDesC8& aString, TBool aCaseInsensitive); 
sl@0
   148
sl@0
   149
	// Closes a string (decrements the reference count and deletes if 0
sl@0
   150
	void Close(RStringTokenEither aString);
sl@0
   151
sl@0
   152
	// Increments the reference count on a string
sl@0
   153
	void IncrementCount(RStringTokenEither aString);
sl@0
   154
sl@0
   155
	inline RStringPool Handle();
sl@0
   156
sl@0
   157
private:	
sl@0
   158
sl@0
   159
	// Check for any undeletable string and delete them now
sl@0
   160
	void DeleteUndeletableStrings(CArrayFixSeg<RStringTokenEither>* aArray[KHashModulo], TInt i);
sl@0
   161
sl@0
   162
private:
sl@0
   163
sl@0
   164
	// Finds a string in the pool.
sl@0
   165
	RStringTokenEither FindDes( const TDesC8& aString, TBool aCaseInsensitive);
sl@0
   166
sl@0
   167
	// Calculates a hash for a descriptor
sl@0
   168
	TUint Hash( const TDesC8& ) const;
sl@0
   169
sl@0
   170
	static TBool CompareCS(const TDesC8& s1, const TDesC8& s2);
sl@0
   171
	static TBool CompareCI(const TDesC8& s1, const TDesC8& s2);
sl@0
   172
sl@0
   173
 private:
sl@0
   174
	// The table.
sl@0
   175
	CArrayFixSeg<RStringTokenEither>* iCSHashTable[KHashModulo];
sl@0
   176
	CArrayFixSeg<RStringTokenEither>* iCIHashTable[KHashModulo];	
sl@0
   177
	RPointerArray<TStringTable> iTablePtrs;	// Stores array of tables, where the index is the table UID
sl@0
   178
	RArray <TStringIdMap> iStringMapList;
sl@0
   179
	RArray <TStringIdMap> iStringMapListReverse;
sl@0
   180
sl@0
   181
	// For rolling back when a leave occurs during CreateTableL
sl@0
   182
	RPointerArray <TStringIdMap> iRollbackMapList;	
sl@0
   183
	RPointerArray <RStringTokenEither> iRollbackHashListCS;
sl@0
   184
	RPointerArray <RStringTokenEither> iRollbackHashListCI;
sl@0
   185
	RPointerArray<MStringPoolCloseCallBack> iCallBacks;
sl@0
   186
	};
sl@0
   187
sl@0
   188
/** 
sl@0
   189
An internal version of the string token class. This class can hold
sl@0
   190
either folding or non-folding versions, and it is up to the user
sl@0
   191
to get it right.
sl@0
   192
@internalComponent
sl@0
   193
*/
sl@0
   194
class RStringTokenEither : public RStringTokenBase
sl@0
   195
	{
sl@0
   196
 public:
sl@0
   197
	inline RStringTokenEither();
sl@0
   198
sl@0
   199
	inline RStringTokenEither(TUint32 aVal);
sl@0
   200
sl@0
   201
	/** Comparison operator
sl@0
   202
		@param aVal The string to compare. */
sl@0
   203
	inline TBool operator==(RStringTokenEither aVal) const;
sl@0
   204
sl@0
   205
	/** Comparison operator
sl@0
   206
		@param aVal The string to compare. */
sl@0
   207
	inline TBool operator!=(RStringTokenEither aVal) const;
sl@0
   208
sl@0
   209
	/** Assignment operator; makes a string token from a string.
sl@0
   210
		@param aVal The string to copy */
sl@0
   211
	inline RStringTokenEither operator=(RStringBase aVal);
sl@0
   212
sl@0
   213
	friend class RStringPool;
sl@0
   214
	friend class RStringEither;
sl@0
   215
	friend class CStringPoolImplementation;
sl@0
   216
	};
sl@0
   217
sl@0
   218
class RStringEither : public RStringBase
sl@0
   219
/**
sl@0
   220
@internalComponent
sl@0
   221
*/
sl@0
   222
	{
sl@0
   223
	public:
sl@0
   224
	RStringEither(CStringPoolImplementation* aPool, RStringTokenEither aVal);
sl@0
   225
	};
sl@0
   226
sl@0
   227
class StringUtils
sl@0
   228
/**
sl@0
   229
@internalComponent
sl@0
   230
*/
sl@0
   231
	{
sl@0
   232
 public:
sl@0
   233
	static inline TBool IsTableEntry(TInt aVal);
sl@0
   234
	static inline TInt TableIndex(TInt aVal);
sl@0
   235
	static inline TInt16 TableUid(TInt aVal);
sl@0
   236
	static inline CStringPoolNode* NodePtr(TInt aVal);
sl@0
   237
	static TInt ValFromIndex(TInt aIndex, TUint16 aTableId);
sl@0
   238
	static TInt ValFromIndexF(TInt aIndex, TUint16 aTableId);
sl@0
   239
	static TInt ValFromIndex(TInt aIndex, TUint16 aTableId, TBool aCaseSensitive);
sl@0
   240
	
sl@0
   241
	static void LogIt(TRefByValue<const TDesC8> aFmt, ...);
sl@0
   242
	static void LogIt1(TRefByValue<const TDesC8> aFmt);
sl@0
   243
	};
sl@0
   244
sl@0
   245
#define _LOGGING
sl@0
   246
sl@0
   247
#if defined (_DEBUG) && defined (_LOGGING)
sl@0
   248
sl@0
   249
/**
sl@0
   250
HTTP Logging macros
sl@0
   251
@internalComponent
sl@0
   252
*/
sl@0
   253
#define __LOG(C)			StringUtils::LogIt1(C);
sl@0
   254
#define __LOG1(C, X)		StringUtils::LogIt(C, X);
sl@0
   255
#define __LOG2(C, X, Y)		StringUtils::LogIt(C, X, Y);
sl@0
   256
#define __LOG3(C, X, Y, Z)	StringUtils::LogIt(C, X, Y, Z);
sl@0
   257
sl@0
   258
#else
sl@0
   259
sl@0
   260
/**
sl@0
   261
NULL macros
sl@0
   262
@internalComponent
sl@0
   263
*/
sl@0
   264
#define __LOG(C)			
sl@0
   265
#define __LOG1(C, X)		
sl@0
   266
#define __LOG2(C, X, Y)		
sl@0
   267
#define __LOG3(C, X, Y, Z)	
sl@0
   268
#endif // !_DEBUG
sl@0
   269
sl@0
   270
sl@0
   271
inline TBool StringUtils::IsTableEntry(TInt aVal)
sl@0
   272
	{
sl@0
   273
	return IS_TABLE_ENTRY(aVal);
sl@0
   274
	}
sl@0
   275
sl@0
   276
inline TInt StringUtils::TableIndex(TInt aVal)
sl@0
   277
	{
sl@0
   278
	return TABLE_INDEX(aVal);
sl@0
   279
	}
sl@0
   280
sl@0
   281
inline TInt16 StringUtils::TableUid(TInt aVal)
sl@0
   282
	{
sl@0
   283
	return (TInt16) TABLE_UID(aVal);
sl@0
   284
	}
sl@0
   285
sl@0
   286
inline CStringPoolNode* StringUtils::NodePtr(TInt aVal)
sl@0
   287
	{
sl@0
   288
	return reinterpret_cast<CStringPoolNode*>(aVal & KTokenToNode);
sl@0
   289
	}
sl@0
   290
sl@0
   291
inline RStringEither::RStringEither(CStringPoolImplementation* aPool, 
sl@0
   292
							 RStringTokenEither aVal)
sl@0
   293
	{
sl@0
   294
	iPool = aPool->Handle();
sl@0
   295
	iVal = aVal.iVal;
sl@0
   296
	}
sl@0
   297
sl@0
   298
inline RStringPool CStringPoolImplementation::Handle()
sl@0
   299
	{
sl@0
   300
	RStringPool p;
sl@0
   301
	p.iImplementation = this;
sl@0
   302
	return p;
sl@0
   303
	}
sl@0
   304
sl@0
   305
inline RStringTokenEither::RStringTokenEither()
sl@0
   306
	{
sl@0
   307
	}
sl@0
   308
sl@0
   309
inline RStringTokenEither::RStringTokenEither(TUint32 aVal)
sl@0
   310
	{
sl@0
   311
	iVal = aVal;
sl@0
   312
	}
sl@0
   313
sl@0
   314
inline TBool RStringTokenEither::operator==(RStringTokenEither aVal) const
sl@0
   315
	{
sl@0
   316
	return iVal == aVal.iVal;
sl@0
   317
	}
sl@0
   318
sl@0
   319
inline TBool RStringTokenEither::operator!=(RStringTokenEither aVal) const
sl@0
   320
	{
sl@0
   321
	return iVal != aVal.iVal;
sl@0
   322
	}
sl@0
   323
sl@0
   324
inline RStringTokenEither RStringTokenEither::operator=(RStringBase aVal)
sl@0
   325
	{
sl@0
   326
	RStringTokenBase b = aVal;
sl@0
   327
	iVal = b.iVal;
sl@0
   328
	return *this;
sl@0
   329
	}
sl@0
   330
sl@0
   331
sl@0
   332
#endif // __STRINGPOOLIMPLEMENTATION_H__