os/graphics/windowing/windowserver/nga/SERVER/INIFILE.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 1996-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
#include <e32std.h>
sl@0
    17
#include "server.h"
sl@0
    18
#include "inifile.h"
sl@0
    19
sl@0
    20
// uncomment so that if the wsini.ini file is not found on drive Wserv loaded from or drive Z: (if different)
sl@0
    21
// it will be searched for on drive C: (if different from Wserv's drive)
sl@0
    22
//#define LOAD_INI_FILE_FROM_DRIVE_Z_OR_C
sl@0
    23
sl@0
    24
sl@0
    25
GLREF_D CDebugLogBase *wsDebugLog;
sl@0
    26
sl@0
    27
_LIT(KDefaultSectionName,"DEFAULT");
sl@0
    28
_LIT(KCommentMarker,"//");
sl@0
    29
_LIT(KScreenSectionName,"SCREEN");
sl@0
    30
const TUint16 KNewSection('[');
sl@0
    31
const TUint16 KSpaceChar(' ');
sl@0
    32
const TUint16 KNewSection2(']');
sl@0
    33
const TInt KDefaultSectionNumber(0);
sl@0
    34
const TInt KDefaultScreenId(-1);
sl@0
    35
sl@0
    36
sl@0
    37
CIniFile::CIniFile()
sl@0
    38
	{}
sl@0
    39
sl@0
    40
CIniFile::~CIniFile()
sl@0
    41
	{
sl@0
    42
	FreeData();
sl@0
    43
	}
sl@0
    44
sl@0
    45
CIniSection * CIniFile::FindSection(TInt aScreen)
sl@0
    46
	{
sl@0
    47
	for (TInt sect = 0; sect < iSectionArray.Count(); ++sect)
sl@0
    48
		{
sl@0
    49
		if (iSectionArray[sect]->Screen() == aScreen)
sl@0
    50
			{
sl@0
    51
			return iSectionArray[sect];
sl@0
    52
			}
sl@0
    53
		}
sl@0
    54
	return NULL;
sl@0
    55
	}
sl@0
    56
sl@0
    57
CIniSection * CIniFile::FindSection(const TDesC& aName)
sl@0
    58
	{
sl@0
    59
	for (TInt sect = 0; sect < iSectionArray.Count(); ++sect)
sl@0
    60
		{
sl@0
    61
		if (iSectionArray[sect]->Screen() == KDefaultScreenId && !iSectionArray[sect]->Name().CompareF(aName))
sl@0
    62
			{
sl@0
    63
			return iSectionArray[sect];
sl@0
    64
			}
sl@0
    65
		}
sl@0
    66
	return NULL;
sl@0
    67
	}
sl@0
    68
sl@0
    69
/* Processes a .ini file entry section name.
sl@0
    70
sl@0
    71
 @return the corresponding index.
sl@0
    72
 @param aLine Input line of text from the .ini file, stripped of comments & excess whitespace.
sl@0
    73
 @leave KErrNoMemory
sl@0
    74
 */
sl@0
    75
CIniSection * CIniFile::AddOrFindIniSectionL(TPtr& aSectionName)
sl@0
    76
	{
sl@0
    77
	aSectionName.Trim();
sl@0
    78
sl@0
    79
	// DEFAULT section
sl@0
    80
	if (aSectionName.CompareF(KDefaultSectionName) == 0)
sl@0
    81
		{ 
sl@0
    82
		return iSectionArray[KDefaultSectionNumber];
sl@0
    83
		}
sl@0
    84
sl@0
    85
	// SCREENx section
sl@0
    86
	if (0 == aSectionName.FindF(KScreenSectionName))
sl@0
    87
		{
sl@0
    88
		TLex lex(aSectionName.Mid(KScreenSectionName().Length()));
sl@0
    89
		TInt screenNum;
sl@0
    90
		lex.SkipSpace();
sl@0
    91
		if (lex.Val(screenNum) == KErrNone)
sl@0
    92
			{
sl@0
    93
			return AddOrFindScreenSectionL(screenNum);
sl@0
    94
			}
sl@0
    95
		}
sl@0
    96
		
sl@0
    97
	// other section
sl@0
    98
	return AddOrFindNamedSectionL(aSectionName);
sl@0
    99
	}
sl@0
   100
sl@0
   101
CIniSection * CIniFile::CreateSectionL(TInt aScreen)
sl@0
   102
	{
sl@0
   103
	CIniSection* newSection = new (ELeave) CIniSection(aScreen);
sl@0
   104
	CleanupStack::PushL( newSection ) ;
sl@0
   105
	newSection->ConstructL() ;
sl@0
   106
	User::LeaveIfError(iSectionArray.Append(newSection));
sl@0
   107
	CleanupStack::Pop( newSection ) ;
sl@0
   108
	if (aScreen + 1 > iScreenCount)
sl@0
   109
		iScreenCount = aScreen + 1;
sl@0
   110
	return newSection;
sl@0
   111
	}
sl@0
   112
sl@0
   113
CIniSection * CIniFile::CreateSectionL(const TDesC& aName)
sl@0
   114
	{
sl@0
   115
	CIniSection* newSection = new (ELeave) CIniSection(KDefaultScreenId);
sl@0
   116
	CleanupStack::PushL( newSection ) ;
sl@0
   117
	newSection->ConstructL(aName) ;
sl@0
   118
	User::LeaveIfError(iSectionArray.Append(newSection));
sl@0
   119
	CleanupStack::Pop( newSection ) ;
sl@0
   120
	return newSection;
sl@0
   121
	}
sl@0
   122
sl@0
   123
sl@0
   124
void CIniFile::doConstructL(RFile &aFile)
sl@0
   125
	{
sl@0
   126
	TFileText textFile;
sl@0
   127
	textFile.Set(aFile);
sl@0
   128
	const TInt KMaxIniLine=256;
sl@0
   129
	TBuf<KMaxIniLine> readLine;
sl@0
   130
	TBool first=ETrue;
sl@0
   131
sl@0
   132
	// always have a [DEFAULT] section
sl@0
   133
	CIniSection * currentSection = CreateSectionL(KDefaultSectionName);
sl@0
   134
sl@0
   135
	FOREVER
sl@0
   136
		{
sl@0
   137
		TInt err=textFile.Read(readLine);
sl@0
   138
		if (err==KErrEof)
sl@0
   139
			break;
sl@0
   140
		User::LeaveIfError(err);
sl@0
   141
sl@0
   142
		if (readLine.Length()>0)
sl@0
   143
			{
sl@0
   144
			if (first && (readLine[0]==0xFFFE || readLine[0]==0xFEFF))
sl@0
   145
				readLine.Delete(0,1);
sl@0
   146
			
sl@0
   147
			// Comment marker "//" indicates the rest of the line should be discarded
sl@0
   148
			TInt commentStart = readLine.Find(KCommentMarker);
sl@0
   149
			if (commentStart != KErrNotFound)
sl@0
   150
				{
sl@0
   151
				readLine.Delete(commentStart, readLine.Length());
sl@0
   152
				}
sl@0
   153
sl@0
   154
			// compact unnecessary whitespace
sl@0
   155
			readLine.TrimAll();
sl@0
   156
sl@0
   157
			// anything left in buffer?
sl@0
   158
			if (readLine.Length() > 0)
sl@0
   159
				{
sl@0
   160
				// section name requires "[" and "]"
sl@0
   161
				if (readLine[0] == KNewSection && readLine.LocateReverse(KNewSection2) == readLine.Length() - 1)
sl@0
   162
					{
sl@0
   163
					TPtr nameText = readLine.MidTPtr(1, readLine.Length() - 2); // strip [ and ]
sl@0
   164
					currentSection = AddOrFindIniSectionL(nameText);
sl@0
   165
					}
sl@0
   166
				else
sl@0
   167
					{
sl@0
   168
					if (currentSection)
sl@0
   169
						{
sl@0
   170
						currentSection->AddVariableL(readLine);
sl@0
   171
						}
sl@0
   172
					}
sl@0
   173
				}
sl@0
   174
			first=EFalse;
sl@0
   175
			}
sl@0
   176
		}
sl@0
   177
sl@0
   178
	if (iScreenCount == 0)
sl@0
   179
		{
sl@0
   180
		iScreenCount = 1;
sl@0
   181
		}
sl@0
   182
	}
sl@0
   183
sl@0
   184
CIniFile* CIniFile::NewL()
sl@0
   185
    {
sl@0
   186
    CIniFile* self = new(ELeave) CIniFile();
sl@0
   187
    CleanupStack::PushL(self);
sl@0
   188
    self->ConstructL();
sl@0
   189
    CleanupStack::Pop(self);
sl@0
   190
    return self;
sl@0
   191
    }
sl@0
   192
sl@0
   193
void CIniFile::FreeData() 
sl@0
   194
	{
sl@0
   195
	iSectionArray.ResetAndDestroy() ;
sl@0
   196
	iScreenCount = 0;
sl@0
   197
	}
sl@0
   198
sl@0
   199
void errFreeData(TAny *aIniFile)
sl@0
   200
	{
sl@0
   201
	((CIniFile *)aIniFile)->FreeData();
sl@0
   202
	}
sl@0
   203
sl@0
   204
HBufC* IniFileSearchPathLC()
sl@0
   205
	{
sl@0
   206
	_LIT(KPath,"\\SYSTEM\\DATA\\");
sl@0
   207
	_LIT(KPathSep,";");
sl@0
   208
	const TInt KLengthPerPath = 2 + KPath().Length();
sl@0
   209
	// work out which drive Wserv loaded from
sl@0
   210
	RProcess self;
sl@0
   211
	TFileName myPath = self.FileName();
sl@0
   212
	TParsePtrC myDrive(myPath);
sl@0
   213
	TDriveUnit myDriveUnit(myDrive.Drive());
sl@0
   214
	// need extra buffer space for search paths for drives Z: or C: ?
sl@0
   215
#if defined(LOAD_INI_FILE_FROM_DRIVE_Z_OR_C)
sl@0
   216
	TInt numPaths = 2;
sl@0
   217
	if (myDriveUnit != EDriveZ && myDriveUnit != EDriveC)
sl@0
   218
		{
sl@0
   219
		numPaths += 1;
sl@0
   220
		}
sl@0
   221
#else
sl@0
   222
	TInt numPaths = 1;
sl@0
   223
	if (myDriveUnit != EDriveZ)
sl@0
   224
		{
sl@0
   225
		numPaths += 1;
sl@0
   226
		}
sl@0
   227
#endif
sl@0
   228
	HBufC* searchPath = HBufC::NewLC(numPaths * KLengthPerPath + (numPaths - 1) * KPathSep().Length());
sl@0
   229
	TPtr pPath(searchPath->Des());
sl@0
   230
	pPath.Append(myDrive.Drive());
sl@0
   231
	pPath.Append(KPath);
sl@0
   232
	if (myDriveUnit != EDriveZ)
sl@0
   233
		{
sl@0
   234
		pPath.Append(KPathSep);
sl@0
   235
		pPath.Append(TDriveUnit(EDriveZ).Name());
sl@0
   236
		pPath.Append(KPath);
sl@0
   237
		}
sl@0
   238
#if defined(LOAD_INI_FILE_FROM_DRIVE_Z_OR_C)
sl@0
   239
	if (myDriveUnit != EDriveC)
sl@0
   240
		{
sl@0
   241
		pPath.Append(KPathSep);
sl@0
   242
		pPath.Append(TDriveUnit(EDriveC).Name());
sl@0
   243
		pPath.Append(KPath);
sl@0
   244
		}
sl@0
   245
#endif
sl@0
   246
	return searchPath;
sl@0
   247
	}
sl@0
   248
sl@0
   249
void CIniFile::ConstructL()
sl@0
   250
	{
sl@0
   251
	TAutoClose<RFs> fs;
sl@0
   252
	User::LeaveIfError(fs.iObj.Connect());
sl@0
   253
	fs.iObj.SetNotifyUser(EFalse);
sl@0
   254
	fs.PushL();
sl@0
   255
	HBufC* searchPath = IniFileSearchPathLC();
sl@0
   256
	_LIT(KFileName,"WSINI.INI");
sl@0
   257
	TFindFile findinifile(fs.iObj);
sl@0
   258
	TInt err=findinifile.FindByPath(KFileName,searchPath);
sl@0
   259
	User::LeaveIfError(err);
sl@0
   260
	CleanupStack::PopAndDestroy(searchPath);
sl@0
   261
 	TAutoClose<RFile> file;
sl@0
   262
	User::LeaveIfError(file.iObj.Open(fs.iObj,findinifile.File(),EFileStreamText|EFileRead));
sl@0
   263
	file.PushL();
sl@0
   264
	CleanupStack::PushL(TCleanupItem(errFreeData,this));
sl@0
   265
	doConstructL(file.iObj);
sl@0
   266
	CleanupStack::Pop(); // TCleanupItem
sl@0
   267
	file.Pop();
sl@0
   268
	fs.Pop();
sl@0
   269
	}
sl@0
   270
sl@0
   271
/* If the Section for the screen exists find the data, otherwise create a new data structure.
sl@0
   272
sl@0
   273
 @param aScreen Screen number
sl@0
   274
 @return index to section
sl@0
   275
 @leave KErrNoMemory
sl@0
   276
 */
sl@0
   277
CIniSection * CIniFile::AddOrFindScreenSectionL(TInt aScreen)
sl@0
   278
	{
sl@0
   279
	CIniSection * section = FindSection(aScreen);
sl@0
   280
	if (!section)
sl@0
   281
		section = CreateSectionL(aScreen);
sl@0
   282
	return section;
sl@0
   283
	}
sl@0
   284
sl@0
   285
/* If the Section exists find the data, otherwise create a new data structure.
sl@0
   286
sl@0
   287
 @param aName section name
sl@0
   288
 @return index to section
sl@0
   289
 @leave KErrNoMemory
sl@0
   290
 */
sl@0
   291
CIniSection * CIniFile::AddOrFindNamedSectionL(const TDesC& aName)
sl@0
   292
	{
sl@0
   293
	CIniSection * section = FindSection(aName);
sl@0
   294
	if (!section)
sl@0
   295
		section = CreateSectionL(aName);
sl@0
   296
	return section;
sl@0
   297
	}
sl@0
   298
sl@0
   299
TBool CIniFile::FindVar(const TDesC &aVarName, TPtrC &aResult)
sl@0
   300
	{
sl@0
   301
	return iSectionArray[KDefaultSectionNumber]->FindVar(aVarName, aResult);
sl@0
   302
	}
sl@0
   303
sl@0
   304
sl@0
   305
TBool CIniFile::FindVar(const TDesC &aVarName, TInt &aResult)
sl@0
   306
	{
sl@0
   307
	return iSectionArray[KDefaultSectionNumber]->FindVar(aVarName, aResult);
sl@0
   308
	}
sl@0
   309
sl@0
   310
TBool CIniFile::FindVar(const TDesC &aVarName)
sl@0
   311
//
sl@0
   312
// Used to simply detect the presence of the specified variable name
sl@0
   313
//
sl@0
   314
	{
sl@0
   315
	TPtrC ptr(NULL,0);
sl@0
   316
	return FindVar(aVarName, ptr);
sl@0
   317
	}
sl@0
   318
sl@0
   319
// FindVar in [SCREENx] sections
sl@0
   320
TBool CIniFile::FindVar( TInt aScreen, const TDesC &aVarName)
sl@0
   321
	{
sl@0
   322
	TPtrC ptr(NULL,0);
sl@0
   323
	return FindVar(aScreen, aVarName, ptr);
sl@0
   324
	}
sl@0
   325
sl@0
   326
TBool CIniFile::FindVar( TInt aScreen, const TDesC& aVarName, TPtrC &aResult )
sl@0
   327
	{
sl@0
   328
	CIniSection * section = FindSection(aScreen);
sl@0
   329
	TBool found = EFalse;
sl@0
   330
	if (section)
sl@0
   331
		found = section->FindVar(aVarName, aResult);
sl@0
   332
	if (!found)
sl@0
   333
		found = FindVar(aVarName, aResult);
sl@0
   334
	return found;
sl@0
   335
	}
sl@0
   336
sl@0
   337
TBool CIniFile::FindVar(TInt aScreen, const TDesC &aVarName, TInt &aResult)
sl@0
   338
	{
sl@0
   339
	CIniSection * section = FindSection(aScreen);
sl@0
   340
	TBool found = EFalse;
sl@0
   341
	if (section)
sl@0
   342
		found = section->FindVar(aVarName, aResult);
sl@0
   343
	if (!found)
sl@0
   344
		found = FindVar(aVarName, aResult);
sl@0
   345
	return found;
sl@0
   346
	}
sl@0
   347
sl@0
   348
// FindVar in named sections
sl@0
   349
TBool CIniFile::FindVar(const TDesC& aSectionName, const TDesC &aVarName)
sl@0
   350
	{
sl@0
   351
	TPtrC ptr(NULL,0);
sl@0
   352
	return FindVar(aSectionName, aVarName, ptr);
sl@0
   353
	}
sl@0
   354
sl@0
   355
TBool CIniFile::FindVar(const TDesC& aSectionName, const TDesC& aVarName, TPtrC &aResult )
sl@0
   356
	{
sl@0
   357
	CIniSection * section = FindSection(aSectionName);
sl@0
   358
	TBool found = EFalse;
sl@0
   359
	if (section)
sl@0
   360
		found = section->FindVar(aVarName, aResult);
sl@0
   361
	if (!found)
sl@0
   362
		found = FindVar(aVarName, aResult);
sl@0
   363
	return found;
sl@0
   364
	}
sl@0
   365
sl@0
   366
TBool CIniFile::FindVar(const TDesC& aSectionName, const TDesC &aVarName, TInt &aResult)
sl@0
   367
	{
sl@0
   368
	CIniSection * section = FindSection(aSectionName);
sl@0
   369
	TBool found = EFalse;
sl@0
   370
	if (section)
sl@0
   371
		found = section->FindVar(aVarName, aResult);
sl@0
   372
	if (!found)
sl@0
   373
		found = FindVar(aVarName, aResult);
sl@0
   374
	return found;
sl@0
   375
	}
sl@0
   376
sl@0
   377
TInt CIniFile::NumberOfScreens() const 
sl@0
   378
	{
sl@0
   379
  	return iScreenCount;
sl@0
   380
	}
sl@0
   381
sl@0
   382
sl@0
   383
// CIniSection.
sl@0
   384
// ini file structure is now in sections like this
sl@0
   385
//
sl@0
   386
// [DEFAULT]
sl@0
   387
// varname value
sl@0
   388
// varname2 value2
sl@0
   389
// [SCREEN0]
sl@0
   390
// screenvar value
sl@0
   391
// etc
sl@0
   392
//
sl@0
   393
// CIniSection represents a section - i.e. section name and content pairs.  
sl@0
   394
// Content pairs are as ini file was previously (so use same code)
sl@0
   395
// [default] section name is optional to support backwards compatibility
sl@0
   396
// if no sutable value is found in a [screenN] section the [default] section will be searched.
sl@0
   397
sl@0
   398
sl@0
   399
CIniSection::CIniSection(TInt aScreen) : iScreen(aScreen)
sl@0
   400
	{}
sl@0
   401
sl@0
   402
void CIniSection::ConstructL()
sl@0
   403
	{
sl@0
   404
	iPtrArray = new (ELeave) CArrayPtrFlat<TDesC>(8) ;
sl@0
   405
	}
sl@0
   406
sl@0
   407
void CIniSection::ConstructL(const TDesC& aName)
sl@0
   408
	{
sl@0
   409
	iName.CreateL(aName);
sl@0
   410
	ConstructL();
sl@0
   411
	}
sl@0
   412
sl@0
   413
CIniSection::~CIniSection()
sl@0
   414
	{
sl@0
   415
	iName.Close();
sl@0
   416
	iPtrArray->ResetAndDestroy() ;
sl@0
   417
	delete iPtrArray ;
sl@0
   418
	}
sl@0
   419
sl@0
   420
inline TInt CIniSection::Screen() const
sl@0
   421
	{ return iScreen; }
sl@0
   422
sl@0
   423
inline const TDesC& CIniSection::Name() const
sl@0
   424
	{ return iName; }
sl@0
   425
sl@0
   426
TBool CIniSection::FindVar( const TDesC& aVarName, TPtrC &aResult )
sl@0
   427
	{
sl@0
   428
	if (iPtrArray)
sl@0
   429
		{
sl@0
   430
		TInt index(KErrNotFound);
sl@0
   431
		
sl@0
   432
		if (FindVarName(aVarName, index))
sl@0
   433
			{
sl@0
   434
			TLex lex((*iPtrArray)[index]->Mid(aVarName.Length()));
sl@0
   435
			lex.SkipSpace();
sl@0
   436
			aResult.Set(lex.Remainder());
sl@0
   437
sl@0
   438
			if (wsDebugLog)
sl@0
   439
				{
sl@0
   440
				wsDebugLog->IniFileSettingRead(iScreen, aVarName, ETrue, aResult);
sl@0
   441
				}
sl@0
   442
			return(ETrue);
sl@0
   443
			}
sl@0
   444
		}
sl@0
   445
sl@0
   446
	if (wsDebugLog)
sl@0
   447
		{
sl@0
   448
		wsDebugLog->IniFileSettingRead(iScreen, aVarName, EFalse, KNullDesC);
sl@0
   449
		}
sl@0
   450
	return(EFalse);
sl@0
   451
	}
sl@0
   452
sl@0
   453
/*
sl@0
   454
 create a TPtrC with just the first word (variable name) in the given string
sl@0
   455
 */
sl@0
   456
TPtrC CIniSection::VarName(const TDesC& aVarString)
sl@0
   457
	{
sl@0
   458
	TInt varLength = aVarString.Locate(KSpaceChar);
sl@0
   459
	if (varLength == KErrNotFound)
sl@0
   460
		{
sl@0
   461
		varLength = aVarString.Length();
sl@0
   462
		}
sl@0
   463
	return aVarString.Left(varLength);
sl@0
   464
	}
sl@0
   465
sl@0
   466
sl@0
   467
TBool CIniSection::FindVar(const TDesC &aVarName, TInt &aResult)
sl@0
   468
	{
sl@0
   469
	TPtrC ptr(NULL,0);
sl@0
   470
	// do text Find
sl@0
   471
	if (FindVar(aVarName, ptr))
sl@0
   472
		{
sl@0
   473
		TLex lex(ptr);
sl@0
   474
		_LIT(HexFormatCheck,"0x");
sl@0
   475
		TPtrC hexPtr(HexFormatCheck);
sl@0
   476
		if(ptr.Left(2) != hexPtr)
sl@0
   477
			{
sl@0
   478
			if (lex.Val(aResult) == KErrNone)
sl@0
   479
				{
sl@0
   480
				return ETrue;
sl@0
   481
				}
sl@0
   482
			}
sl@0
   483
		else
sl@0
   484
			{
sl@0
   485
			lex.SkipAndMark(2); //To skip 0x in hex code
sl@0
   486
			if (lex.Val((TUint32&)aResult, EHex) == KErrNone)
sl@0
   487
				{
sl@0
   488
				return ETrue;
sl@0
   489
				}
sl@0
   490
			}
sl@0
   491
		}
sl@0
   492
sl@0
   493
	return EFalse;
sl@0
   494
	}
sl@0
   495
sl@0
   496
sl@0
   497
/*
sl@0
   498
 Find variable name in sorted array, using binary search
sl@0
   499
 @param aVarName variable name to search for, must have any space and variable value stripped.
sl@0
   500
 @param aIndex output index of matching or preceeding item
sl@0
   501
 @return
sl@0
   502
 */
sl@0
   503
TBool CIniSection::FindVarName(const TDesC& aVarName, TInt& aIndex)
sl@0
   504
	{
sl@0
   505
	// Binary Search
sl@0
   506
	// left is lowest index to include, right is highest index + 1;
sl@0
   507
	TInt left = 0;
sl@0
   508
	TInt right = iPtrArray->Count();
sl@0
   509
sl@0
   510
	while (right > left)
sl@0
   511
		{
sl@0
   512
		TInt middle = (left + right)>>1;
sl@0
   513
		// compare to start of variable string
sl@0
   514
		TPtrC cmpString = VarName(*(*iPtrArray)[middle]);
sl@0
   515
		TInt cmp = aVarName.CompareF(cmpString);
sl@0
   516
sl@0
   517
		if (cmp == 0)
sl@0
   518
			{
sl@0
   519
			aIndex = middle;
sl@0
   520
			return ETrue;
sl@0
   521
			}
sl@0
   522
		else if (cmp > 0)
sl@0
   523
			{
sl@0
   524
			left = middle + 1;
sl@0
   525
			}
sl@0
   526
		else
sl@0
   527
			{
sl@0
   528
			right = middle;
sl@0
   529
			}
sl@0
   530
		}
sl@0
   531
sl@0
   532
	aIndex = right;
sl@0
   533
	return EFalse;
sl@0
   534
	}
sl@0
   535
sl@0
   536
void CIniSection::AddVariableL(const TDesC& aNewVariable)
sl@0
   537
	{
sl@0
   538
	// use variable name only for search
sl@0
   539
	TPtrC varName = VarName(aNewVariable);
sl@0
   540
	TInt index(0);
sl@0
   541
	
sl@0
   542
	// ignore duplicate definitions
sl@0
   543
	if (0 == FindVarName(varName, index))
sl@0
   544
		{ // insert in sorted array
sl@0
   545
		HBufC* hbuf = aNewVariable.AllocLC() ;
sl@0
   546
		iPtrArray->InsertL(index, hbuf);
sl@0
   547
		CleanupStack::Pop(hbuf);
sl@0
   548
		}
sl@0
   549
	}
sl@0
   550