os/graphics/windowing/windowserver/test/t_genericplugin/src/t_cinidata.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2008-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 /**
    17  @file
    18  @test
    19  @internalComponent
    20 */
    21 
    22 #include "t_cinidata.h"
    23 #include <f32file.h>
    24 #include <e32std.h>
    25 
    26 // Default directory to look for INI file
    27 _LIT(KIniFileDir,"Z:\\System\\Data\\");
    28 
    29 // Constant Value changed for DEF047130 fix
    30 const TInt KTokenSize = 256;
    31 
    32 enum TIniPanic
    33 	{
    34 	ESectionNameTooBig,
    35 	EKeyNameTooBig,
    36 	};
    37 
    38 //
    39 void Panic(TIniPanic aPanic)
    40 	{
    41 	_LIT(CIniData,"CIniData");
    42 	User::Panic(CIniData,aPanic);
    43 	}
    44 
    45 /**
    46 Constructor
    47 */
    48 CIniData::CIniData() : iPtr(NULL,0)
    49 	{
    50 	__DECLARE_NAME(_S("CIniData"));
    51 	}
    52 
    53 /**
    54 Destructor.
    55 Frees the resources located in second-phase constructor
    56 */
    57 CIniData::~CIniData()
    58 	{
    59 	delete (TText*)iPtr.Ptr();
    60 	delete iToken;
    61 	delete iName;
    62 	}
    63 
    64 /**
    65  Creates, and returns a pointer to CIniData object, leave on failure
    66  @param aName the name of the file which contains the ini data
    67  @param aDelimeter	Delimiter used in wsini.ini file between variable and value
    68  @return A pointer to the CiniData object
    69 */
    70 CIniData* CIniData::NewL(const TDesC& aName, char aDelimiter)
    71 	{
    72 	CIniData* p = new(ELeave) CIniData;
    73 	CleanupStack::PushL(p);
    74 	p->ConstructL(aName, aDelimiter);
    75 	CleanupStack::Pop();
    76 	return p;
    77 	}
    78 
    79 /**
    80  Second-phase constructor.
    81  The function attempts to allocate a buffer and Read file's contents into iPtr
    82  @param aName the name of the file which contains the ini data
    83  @param aDelimeter	Delimiter used in wsini.ini file between variable and value
    84  @leave One of the system-wide error codes
    85 */
    86 void CIniData::ConstructL(const TDesC& aName, char aDelimiter)
    87 	{
    88 	//Set delimiter
    89 	iDelimiter = aDelimiter;
    90 	
    91  	// Allocate space for token
    92 	iToken=HBufC::NewL(KTokenSize+2);	// 2 extra chars for [tokenName]
    93 
    94 	// Connect to file server
    95 	TAutoClose<RFs> fs;
    96 	User::LeaveIfError(fs.iObj.Connect());
    97 	fs.PushL();
    98 
    99 	// Find file, given name
   100 	TFindFile ff(fs.iObj);
   101 	User::LeaveIfError(ff.FindByDir(aName, KIniFileDir));
   102 	iName = ff.File().AllocL();
   103 
   104 	// Open file
   105 	TAutoClose<RFile> file;
   106 	TInt size;
   107 	User::LeaveIfError(file.iObj.Open(fs.iObj,*iName,EFileStreamText|EFileShareReadersOrWriters));
   108 	file.PushL();
   109 
   110 	// Get file size and read in
   111 	User::LeaveIfError(file.iObj.Size(size));
   112 	TText* data = (TText*)User::AllocL(size);
   113 	iPtr.Set(data, size/sizeof(TText), size/sizeof(TText));
   114 	TPtr8 dest((TUint8*)data, 0, size);
   115 	User::LeaveIfError(file.iObj.Read(dest));
   116 	TUint8* ptr = (TUint8*)data;
   117 
   118 	//
   119 	// This is orderred as FEFF assuming the processor is Little Endian
   120 	// The data in the file is FFFE.		PRR 28/9/98
   121 	//
   122 	if(size>=(TInt)sizeof(TText) && iPtr[0]==0xFEFF)
   123 	{
   124 		// UNICODE Text file so lose the FFFE
   125 		Mem::Copy(ptr, ptr+sizeof(TText), size-sizeof(TText));
   126 		iPtr.Set(data, size/sizeof(TText)-1, size/sizeof(TText)-1);
   127 	}
   128 	else if(size)
   129 	{
   130 		// NON-UNICODE so convert to UNICODE
   131 		TText* newdata = (TText*)User::AllocL(size*sizeof(TText));
   132 		iPtr.Set(newdata, size, size);
   133 		TInt i;
   134 		for(i = 0 ; i<size ; ++i)
   135 			iPtr[i] = ptr[i];
   136 		delete data;
   137 	}
   138 
   139 	file.Pop();
   140 	fs.Pop();
   141 }
   142 
   143 /**
   144 Find a text value from given aKeyName regardless the section in the ini data file
   145 @param aKeyName Key being searched for
   146 @param aResult On return, contains the text result 
   147 @return ETrue if found, otherwise EFalse
   148 */
   149 TBool CIniData::FindVar(const TDesC& aKeyName, TPtrC& aResult)
   150 	{
   151 	// Call with no section, so starts at beginning
   152 	if (FindVar((TDesC&)KNullDesC , aKeyName, aResult))
   153 		return(ETrue);
   154 	else
   155 		return(EFalse);
   156 	}
   157 
   158 /**
   159 Find a text value from given aKeyName and aSecName in the ini data file
   160 @param aSectName Section containing key
   161 @param aKeyName Key being searched for in aSectName
   162 @param aResult On return, contains the text result 
   163 @return ETrue if found, otherwise EFalse
   164 */
   165 TBool CIniData::FindVar(const TDesC& aSectName,const TDesC& aKeyName,TPtrC& aResult)
   166 	{
   167 
   168 	__ASSERT_DEBUG(aSectName.Length()<=KTokenSize,Panic(ESectionNameTooBig));
   169 	__ASSERT_DEBUG(aKeyName.Length()<=KTokenSize,Panic(EKeyNameTooBig));
   170 
   171 	TInt posStartOfSection(0);
   172 	TInt posEndOfSection(iPtr.Length()); // Default to the entire length of the ini data
   173 	TPtrC SearchBuf;
   174 
   175 	// If we have a section, set pos to section start
   176 	TInt posI(0);	// Position in internal data Buffer
   177 	if( aSectName.Length() > 0 )
   178 		{
   179 		TBool FoundSection(false);
   180 		while ( ! FoundSection )
   181 			{
   182 			// Move search buffer to next area of interest
   183 			SearchBuf.Set(iPtr.Mid(posI));
   184 
   185 			// Make up token "[SECTIONNAME]", to search for
   186 			TPtr sectionToken = iToken->Des();
   187 			_LIT(sectionTokenFmtString,"[%S]");
   188 			sectionToken.Format(sectionTokenFmtString,&aSectName);
   189 
   190 			// Search for next occurrence of aSectName
   191 			TInt posSB = SearchBuf.Find(sectionToken);
   192 
   193 			if (posSB == KErrNotFound)
   194 				return(EFalse);
   195 
   196 			// Check this is at beginning of line (ie. non-commented)
   197 			// ie. Check preceding char was LF
   198 			if(posSB>0)
   199 				{
   200 				// Create a Buffer, starting one char before found subBuf
   201 				TPtrC CharBefore(SearchBuf.Right(SearchBuf.Length()-posSB+1));
   202 				// Check first char is end of prev
   203 				if(CharBefore[0] == '\n')
   204 					{
   205 					FoundSection = ETrue;		// found
   206 					posI = posI + posSB;
   207 					}
   208 				else
   209 					{
   210 					posI = posI + posSB + 1;	// try again
   211 					}
   212 				}
   213 			else
   214 				{
   215 				FoundSection = ETrue;
   216 				}
   217 
   218 			}	// while ( ! FoundSection ) 
   219 
   220 		// Set start of section, after section name, (incl '[' and ']')
   221 		posStartOfSection = posI + aSectName.Length() + 2;
   222 
   223 		// Set end of section, by finding begin of next section or end
   224 		SearchBuf.Set(iPtr.Mid(posI));
   225 		_LIT(nextSectionBuf,"\n[");
   226 		TInt posSB = SearchBuf.Find(nextSectionBuf);
   227 		if(posSB != KErrNotFound)
   228 			{
   229 			posEndOfSection = posI + posSB;
   230 			}
   231 		else
   232 			{
   233 			posEndOfSection = iPtr.Length();
   234 			}
   235 
   236 		}	// if( aSectName.Length() > 0 )
   237 
   238 	// Look for key in ini file data Buffer
   239 	posI = posStartOfSection;
   240 	TBool FoundKey(false);
   241 	while ( ! FoundKey )
   242 		{
   243 		// Search for next occurrence of aKeyName
   244 		SearchBuf.Set(iPtr.Mid(posI,posEndOfSection-posI));
   245 		TInt posSB = SearchBuf.Find(aKeyName);
   246 
   247 		// If not found, return
   248 		if (posSB == KErrNotFound)
   249 			return(EFalse);
   250 
   251 		// Check this is at beginning of line (ie. non-commented)
   252 		// ie. Check preceding char was CR or LF
   253 		if(posSB>0)
   254 			{
   255 			// Create a Buffer, starting one char before found subBuf
   256 			TPtrC CharBefore(SearchBuf.Right(SearchBuf.Length()-posSB+1));
   257 			// Check if the first char is end of prev and also check 
   258 			// if the token found is not a substring of another string  
   259 			TBool beginningOK = ((CharBefore[0] == '\n') || (CharBefore[0] == ' ') || (CharBefore[0] == '\t'));
   260 			TBool endingOK = ((CharBefore[aKeyName.Length()+1] == iDelimiter) || (CharBefore[aKeyName.Length()+1] == ' ') || (CharBefore[aKeyName.Length()+1] == '\t'));
   261 			if (beginningOK && endingOK)
   262 				{
   263 				FoundKey = ETrue;
   264 				posI = posI + posSB;
   265 				}
   266 			else
   267 				{
   268 				posI = posI + posSB + 1;
   269 				}
   270 			}
   271 		else
   272 			{
   273 			FoundKey = ETrue;
   274 			}
   275 		}	// while ( ! FoundKey )
   276 
   277 	// Set pos, to just after iDelimiter sign
   278 	SearchBuf.Set(iPtr.Mid(posI));
   279 	TInt posSB = SearchBuf.Locate(iDelimiter);
   280 	if(posSB==KErrNotFound)		// Illegal format, should flag this...
   281 		return(EFalse);
   282 
   283 	// Identify start and end of data (EOL or EOF)
   284 	posI = posI + posSB + 1;	// 1 char after iDelimiter
   285 	TInt posValStart = posI;
   286 	TInt posValEnd;
   287 	SearchBuf.Set(iPtr.Mid(posI));
   288 	posSB = SearchBuf.Locate('\r');
   289 	if(posSB != KErrNotFound)
   290 		{
   291 		posValEnd = posI + posSB;
   292 		}
   293 	else
   294 		{
   295 		posValEnd = iPtr.Length();
   296 		}
   297 
   298 	// Check we are still in the section requested
   299 	if( aSectName.Length() > 0 )
   300 		{
   301 		if( posValEnd > posEndOfSection )
   302 			{
   303 			return(EFalse);
   304 			}
   305 		}
   306 	// Parse Buffer from posn of key
   307 	// Start one space after '='
   308 	TLex lex(iPtr.Mid(posValStart, posValEnd-posValStart));
   309 	lex.SkipSpaceAndMark();		// Should be at the start of the data
   310 	aResult.Set(lex.MarkedToken().Ptr(),posValEnd-posValStart - lex.Offset() );
   311 	return(ETrue);
   312 	}
   313 
   314 /**
   315 Find an integer value from given aKeyName regardless the section in the ini data file
   316 @param aKeyName Key being searched for
   317 @param aResult On return, contains the TInt result 
   318 @return ETrue if found, otherwise EFalse
   319 */
   320 TBool CIniData::FindVar(const TDesC& aKeyName, TInt& aResult)
   321 	{
   322 	TPtrC ptr(NULL,0);
   323 	if (FindVar(aKeyName,ptr))
   324 		{
   325 		TLex lex(ptr);
   326 		if (lex.Val(aResult) == KErrNone)
   327 			return(ETrue);
   328 		}
   329 	return(EFalse);
   330 	}
   331 
   332 /**
   333 Find an integer value from given aKeyName and aSecName in the ini data file
   334 @param aSectName Section containing key
   335 @param aKeyName Key being searched for  in aSectName
   336 @param aResult On return, contains TInt result 
   337 @return ETrue if found, otherwise EFalse
   338 */
   339 TBool CIniData::FindVar(const TDesC& aSection,const TDesC& aKeyName,TInt& aResult)
   340 	{
   341 	TPtrC ptr(NULL,0);
   342 	if (FindVar(aSection,aKeyName,ptr))
   343 		{
   344 		TLex lex(ptr);
   345 		if (lex.Val(aResult) == KErrNone)
   346 			return(ETrue);
   347 		}
   348 	return(EFalse);
   349 }