os/ossrv/lowlevellibsandfws/apputils/src/BaArchiveImpl.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) 2004-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 <barsc2.h>
sl@0
    17
#include "BaArchiveImpl.h"
sl@0
    18
#include <baflpan.h>
sl@0
    19
#include <bautils.h>
sl@0
    20
sl@0
    21
/** TCleanupItem function.
sl@0
    22
@internalComponent
sl@0
    23
*/
sl@0
    24
LOCAL_C void CloseAndDeleteFile(TAny* aFile)
sl@0
    25
	{
sl@0
    26
	RFile* const file=static_cast<RFile*>(aFile);
sl@0
    27
	if (file!=NULL)
sl@0
    28
		{
sl@0
    29
		file->Close();
sl@0
    30
		delete file;
sl@0
    31
		}
sl@0
    32
	}
sl@0
    33
sl@0
    34
/** TCleanupItem function.
sl@0
    35
@internalComponent
sl@0
    36
*/
sl@0
    37
LOCAL_C void CloseAndDeleteChunk(TAny* aChunk)
sl@0
    38
	{
sl@0
    39
	RChunk* const rchunk=static_cast<RChunk*>(aChunk);
sl@0
    40
	if (rchunk!=NULL)
sl@0
    41
		{
sl@0
    42
		rchunk->Close();
sl@0
    43
		delete rchunk;
sl@0
    44
		}
sl@0
    45
	}
sl@0
    46
sl@0
    47
/** Default Constructor
sl@0
    48
@internalComponent
sl@0
    49
*/
sl@0
    50
CResourceArchiveImpl::CResourceArchiveImpl() :
sl@0
    51
	iCurrentIndex(0),
sl@0
    52
	iSpiFileType(KNullUid)
sl@0
    53
	{}
sl@0
    54
sl@0
    55
/** Class destructor
sl@0
    56
@internalComponent 
sl@0
    57
*/
sl@0
    58
CResourceArchiveImpl::~CResourceArchiveImpl()
sl@0
    59
	{
sl@0
    60
	iRscList.Close();
sl@0
    61
	
sl@0
    62
	for(TInt i=0; i<iSpiChunkBufferArray.Count(); i++)
sl@0
    63
		{
sl@0
    64
		RChunk* const chunk = iSpiChunkBufferArray[i];                     
sl@0
    65
		chunk->Close();
sl@0
    66
		delete chunk;
sl@0
    67
		}
sl@0
    68
	
sl@0
    69
	iSpiChunkBufferArray.Reset();
sl@0
    70
	iSpiChunkBufferArray.Close();
sl@0
    71
sl@0
    72
	iSpiBufferArray.ResetAndDestroy();
sl@0
    73
	iSpiBufferArray.Close();
sl@0
    74
	iCurrentIndex=0;
sl@0
    75
	iSpiFileType=KNullUid;
sl@0
    76
#ifdef __BASPITEST__
sl@0
    77
	iSpiFileNameArray.Reset();
sl@0
    78
	iSpiFileNameArray.Close();
sl@0
    79
#endif	
sl@0
    80
	}
sl@0
    81
sl@0
    82
/** Creates a CResourceArchiveImpl instance to read a single spi file
sl@0
    83
@internalComponent
sl@0
    84
*/
sl@0
    85
CResourceArchiveImpl* CResourceArchiveImpl::NewL(RFs& aFs,const TDesC& aName)
sl@0
    86
	{
sl@0
    87
	CResourceArchiveImpl* self=new (ELeave) CResourceArchiveImpl;
sl@0
    88
	CleanupStack::PushL(self);
sl@0
    89
	self->ConstructL(aFs,aName);
sl@0
    90
	CleanupStack::Pop();
sl@0
    91
	return self;
sl@0
    92
	}
sl@0
    93
sl@0
    94
/** Creates a CResourceArchiveImpl instance to read a set of spi files
sl@0
    95
@internalComponent
sl@0
    96
*/	
sl@0
    97
CResourceArchiveImpl* CResourceArchiveImpl::NewL(RFs& aFs,const TDesC& aPath,const TDesC& aPattern)
sl@0
    98
	{
sl@0
    99
	CResourceArchiveImpl* self=new (ELeave) CResourceArchiveImpl;
sl@0
   100
	CleanupStack::PushL(self);
sl@0
   101
	self->ConstructL(aFs,aPath,aPattern);
sl@0
   102
	CleanupStack::Pop();
sl@0
   103
	return self;	
sl@0
   104
	}
sl@0
   105
sl@0
   106
/** ConstructL method of CResourceArchiveImpl
sl@0
   107
@internalComponent
sl@0
   108
*/
sl@0
   109
void CResourceArchiveImpl::ConstructL(RFs& aFs,const TDesC& aName)
sl@0
   110
	{
sl@0
   111
	RArray<TPtrC8> hiddenRscList;
sl@0
   112
	CleanupClosePushL(hiddenRscList);
sl@0
   113
	//Open the file and initialise the buffer pointer
sl@0
   114
	TPtr8 bufferPtr(NULL,0,0);
sl@0
   115
	OpenFileL(aFs,aName,bufferPtr);
sl@0
   116
	//Now validate the header
sl@0
   117
	ValidateHeaderL(bufferPtr);
sl@0
   118
	//Process all the rsc entry
sl@0
   119
	ProcessEntryL(bufferPtr,hiddenRscList);	
sl@0
   120
	CleanupStack::PopAndDestroy();
sl@0
   121
	}
sl@0
   122
	
sl@0
   123
sl@0
   124
/** Function to get the string representation of a language code
sl@0
   125
assume language code has max N digits currently equal to ELangMaximum
sl@0
   126
note that the minimum suffix required is NN so TLang(0-9) needs to have a 0
sl@0
   127
e.g s01...s09, s10, s100 etc
sl@0
   128
@internalComponent
sl@0
   129
*/
sl@0
   130
static void GetLangCodeStringRep(TLanguage lang,TDes& aStringRep)
sl@0
   131
{
sl@0
   132
	aStringRep.Zero();
sl@0
   133
	//special case for 0-9 where you need the 0 appended, langcode min two digit
sl@0
   134
	if (lang<10)
sl@0
   135
		aStringRep.AppendNumFixedWidthUC(lang,EDecimal,2);
sl@0
   136
	else
sl@0
   137
		aStringRep.AppendFormat(_L("%d"),lang);
sl@0
   138
}
sl@0
   139
sl@0
   140
/**
sl@0
   141
Open the file and initialise the aBufferPtr to point at the buffer
sl@0
   142
@internalComponent
sl@0
   143
*/
sl@0
   144
void CResourceArchiveImpl::OpenFileL(RFs& aFs,const TDesC& aName,TPtr8& aBufferPtr)
sl@0
   145
	{
sl@0
   146
	#ifdef __WINS__
sl@0
   147
	TUint8* romAddress=NULL;
sl@0
   148
	#else // ! __WINS__
sl@0
   149
	TUint8* romAddress=aFs.IsFileInRom(aName);
sl@0
   150
	#endif // ! __WINS__
sl@0
   151
	
sl@0
   152
	RFile* const file=new(ELeave) RFile;
sl@0
   153
	CleanupStack::PushL(TCleanupItem(CloseAndDeleteFile,file));
sl@0
   154
	User::LeaveIfError(file->Open(aFs, aName, EFileStream | EFileRead | EFileShareReadersOnly));
sl@0
   155
	TInt fileSize=0;
sl@0
   156
	User::LeaveIfError(file->Size(fileSize));						
sl@0
   157
	if (romAddress)
sl@0
   158
		{	
sl@0
   159
		aBufferPtr.Set(romAddress,fileSize,fileSize);
sl@0
   160
		}
sl@0
   161
	else
sl@0
   162
		{
sl@0
   163
		RChunk* rchunk = new(ELeave) RChunk;
sl@0
   164
		
sl@0
   165
		CleanupStack::PushL(TCleanupItem(CloseAndDeleteChunk,rchunk));
sl@0
   166
		
sl@0
   167
		TInt createRet = rchunk->CreateDisconnectedLocal(0, 0, fileSize);
sl@0
   168
		TInt commitRet = rchunk->Commit(0, fileSize);
sl@0
   169
		
sl@0
   170
		// Create a chunk to store the large file contents of which is closed and destroyed in destructor...
sl@0
   171
		if(createRet==KErrNone && commitRet==KErrNone)
sl@0
   172
			{
sl@0
   173
			TUint8* chunkBase = (TUint8*)rchunk->Base();
sl@0
   174
			TPtr8 tempBuffer(chunkBase, fileSize);
sl@0
   175
			User::LeaveIfError(file->Read(0, tempBuffer, fileSize));
sl@0
   176
			aBufferPtr.Set(tempBuffer);
sl@0
   177
			iSpiChunkBufferArray.AppendL(rchunk);
sl@0
   178
			
sl@0
   179
			CleanupStack::Pop(rchunk);
sl@0
   180
			}
sl@0
   181
		else	// Unable to create a chunk so use heap memory...
sl@0
   182
			{
sl@0
   183
			HBufC8* fileBuffer=HBufC8::NewMaxLC(fileSize);
sl@0
   184
			aBufferPtr.Set(fileBuffer->Des());
sl@0
   185
			User::LeaveIfError(file->Read(0,aBufferPtr,fileSize));
sl@0
   186
			iSpiBufferArray.AppendL(fileBuffer);
sl@0
   187
			
sl@0
   188
			CleanupStack::Pop(fileBuffer);
sl@0
   189
			CleanupStack::PopAndDestroy(rchunk);
sl@0
   190
			}
sl@0
   191
		}
sl@0
   192
	//can close the file now
sl@0
   193
	CleanupStack::PopAndDestroy(file);
sl@0
   194
#ifdef __BASPITEST__
sl@0
   195
	iSpiFileNameArray.AppendL(aName);
sl@0
   196
#endif		
sl@0
   197
	}
sl@0
   198
sl@0
   199
/**
sl@0
   200
Validate the spi header in the file buffer
sl@0
   201
On return it will update the bufferPtr to point to start of the first
sl@0
   202
rsc entry
sl@0
   203
@internalComponent
sl@0
   204
*/
sl@0
   205
void CResourceArchiveImpl::ValidateHeaderL(TPtr8& aBufferPtr)
sl@0
   206
	{
sl@0
   207
	//Getting the 32 bytes header information. At the moment maybe we should just buffer 16 bytes
sl@0
   208
	//of the header as the remaining 16 bytes are padding bytes
sl@0
   209
	TUidType uidType=TCheckedUid(aBufferPtr.Left(16)).UidType();			
sl@0
   210
	if (uidType[0]!=KSpiFileUid)
sl@0
   211
		User::Leave(KErrCorrupt);
sl@0
   212
	//now get the spi file type
sl@0
   213
	TUid spiFileType=TUid::Uid(uidType[1].iUid);
sl@0
   214
	if (iSpiFileType==KNullUid)
sl@0
   215
		iSpiFileType=spiFileType;
sl@0
   216
	//also check consistency with previous spi files
sl@0
   217
	__ASSERT_DEBUG(iSpiFileType==spiFileType,::Panic(EBafPanicBadResourceFileFormat));							
sl@0
   218
sl@0
   219
	//update the bufferPtr to point to start of first rsc entry
sl@0
   220
	aBufferPtr.Set(aBufferPtr.MidTPtr(KSpiFirstRscOffset));	
sl@0
   221
	}
sl@0
   222
sl@0
   223
/**
sl@0
   224
Process all the entry found in the buffer and update the internal rscList
sl@0
   225
@internalComponent
sl@0
   226
*/
sl@0
   227
void CResourceArchiveImpl::ProcessEntryL(TPtr8& aBufferPtr,RArray<TPtrC8>& aHiddenList)
sl@0
   228
	{
sl@0
   229
	//now traverse content of the spi file and build up the hidden list and the TRscEntry array
sl@0
   230
	TRscEntry rscEntry;
sl@0
   231
	while (aBufferPtr.Length()>0)
sl@0
   232
		{
sl@0
   233
		//length(first 4 bytes) and the actual rsc file size(second 4 bytes)
sl@0
   234
		//Retrieving the rscfilename length
sl@0
   235
		TUint32 rscFileNameLength=LittleEndianFourByteInteger(aBufferPtr,0);
sl@0
   236
		TUint32 rscFileSize=LittleEndianFourByteInteger(aBufferPtr,4);
sl@0
   237
		TUint32 paddingbyte=(4-((rscFileNameLength+rscFileSize)%4))%4;
sl@0
   238
		__ASSERT_DEBUG((rscFileNameLength+rscFileSize+paddingbyte)%4==0,::Panic(EBafPanicFileSize));
sl@0
   239
		//construct the TRscEntry				
sl@0
   240
		rscEntry.iRscName.Set(aBufferPtr.Mid(8,rscFileNameLength));
sl@0
   241
		rscEntry.iRscData.Set(aBufferPtr.Mid(8+rscFileNameLength,rscFileSize));
sl@0
   242
#ifdef __BASPITEST__
sl@0
   243
		rscEntry.iFileNamePtr.Set(iSpiFileNameArray[iSpiFileNameArray.Count()-1].Mid(0));
sl@0
   244
#endif				
sl@0
   245
		//update the buffer pointer
sl@0
   246
		aBufferPtr.Set(aBufferPtr.MidTPtr(8+rscFileNameLength+rscFileSize+paddingbyte));
sl@0
   247
sl@0
   248
		//process the TRscEntry
sl@0
   249
		if (rscEntry.iRscData.Length()==0)
sl@0
   250
			aHiddenList.AppendL(rscEntry.iRscName);
sl@0
   251
		else
sl@0
   252
			{
sl@0
   253
			//if can find a matching resource name entry in the hidden list ignore this entry
sl@0
   254
			TIdentityRelation<TPtrC8> identity(MatchDescriptor);
sl@0
   255
			if (aHiddenList.Find(rscEntry.iRscName,identity)==KErrNotFound)
sl@0
   256
				{
sl@0
   257
				//note no duplicate entry, this implies the lastly mounted resource
sl@0
   258
				//entry is preferred over earlier mounted resource entry.(REPLACING)
sl@0
   259
				TInt ret=iRscList.InsertInOrder(rscEntry,EntryOrder);
sl@0
   260
				if (ret!=KErrNone && ret!=KErrAlreadyExists)
sl@0
   261
					User::Leave(ret);
sl@0
   262
				}		
sl@0
   263
			}
sl@0
   264
		}//end while loop	
sl@0
   265
	}
sl@0
   266
	
sl@0
   267
/** ConstructL method that accepts a spi path and a matching pattern
sl@0
   268
aPath here must end with \\ as well e.g. z:\\private\\10009d8f\\ 
sl@0
   269
aPattern should be the spi file withou rom image id and extension
sl@0
   270
@internalComponent
sl@0
   271
*/
sl@0
   272
void CResourceArchiveImpl::ConstructL(RFs& aFs,const TDesC& aPath,const TDesC& aPattern)
sl@0
   273
	{
sl@0
   274
	//Get the downgradepath
sl@0
   275
	RArray<TLanguage> downgradeList;
sl@0
   276
	BaflUtils::GetDowngradePathL(aFs,User::Language(),downgradeList);
sl@0
   277
	CleanupClosePushL(downgradeList);
sl@0
   278
	
sl@0
   279
	//used to check if there is any matching pattern and leave with KErrNotFound if
sl@0
   280
	//no matching spi
sl@0
   281
	TInt spiDiscovered=0;
sl@0
   282
	
sl@0
   283
	//sort out the language specific spi first
sl@0
   284
	for (int i=0;i<=downgradeList.Count();i++)	
sl@0
   285
		{
sl@0
   286
		//list all the files in the directory that matches this language
sl@0
   287
		TFileName matchPattern;
sl@0
   288
		TBuf<5> langExtension;
sl@0
   289
		if (i<downgradeList.Count())
sl@0
   290
			GetLangCodeStringRep(downgradeList[i],langExtension);
sl@0
   291
		else
sl@0
   292
			{
sl@0
   293
			//process the default spi lastly
sl@0
   294
			langExtension.Append(_L("pi"));			
sl@0
   295
			}
sl@0
   296
		matchPattern.AppendFormat(_L("%S%S-*-*.s%S"),&aPath,&aPattern,&langExtension);
sl@0
   297
sl@0
   298
		//Get all the spis that match the language pattern
sl@0
   299
		CDir* spiList=NULL;
sl@0
   300
		User::LeaveIfError(aFs.GetDir(matchPattern,KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive,ESortByName,spiList));
sl@0
   301
		CleanupStack::PushL(spiList);
sl@0
   302
sl@0
   303
		//the hidden list is only a temporary variable for each language
sl@0
   304
		RArray<TPtrC8> hiddenRscList;
sl@0
   305
		CleanupClosePushL(hiddenRscList);
sl@0
   306
		
sl@0
   307
		//reverse processing
sl@0
   308
		for (TInt i=spiList->Count()-1;i>=0;i--)
sl@0
   309
			{
sl@0
   310
			//reuse the matchPattern buffer to hold the full path of the spi file name
sl@0
   311
			matchPattern.Zero();
sl@0
   312
			matchPattern.Append(aPath);
sl@0
   313
			matchPattern.Append((*spiList)[i].iName);
sl@0
   314
sl@0
   315
			//Open the file and initialise the buffer pointer
sl@0
   316
			TPtr8 bufferPtr(NULL,0,0);
sl@0
   317
			OpenFileL(aFs,matchPattern,bufferPtr);
sl@0
   318
	
sl@0
   319
			//Now validate the header
sl@0
   320
			ValidateHeaderL(bufferPtr);
sl@0
   321
	
sl@0
   322
			//Process all the rsc entry
sl@0
   323
			ProcessEntryL(bufferPtr,hiddenRscList);
sl@0
   324
			}//end for loop
sl@0
   325
		spiDiscovered+=spiList->Count();
sl@0
   326
		CleanupStack::PopAndDestroy(2);				//hiddenRscList and spiList
sl@0
   327
		}//end for loop
sl@0
   328
	//if no spi files are discovered at all leave with KErrNotFound
sl@0
   329
	if (spiDiscovered==0)
sl@0
   330
		User::Leave(KErrNotFound);
sl@0
   331
	CleanupStack::PopAndDestroy(&downgradeList);
sl@0
   332
	}
sl@0
   333
sl@0
   334
/** Create the next CResourceFile that correponds to the next resource file
sl@0
   335
in the resource archive(SPI) file
sl@0
   336
@internalComponent
sl@0
   337
*/
sl@0
   338
CResourceFile* CResourceArchiveImpl::NextL(HBufC*& aRscFileName)
sl@0
   339
	{
sl@0
   340
	//if end of list return nothing
sl@0
   341
	if (iCurrentIndex==iRscList.Count())
sl@0
   342
		return NULL;
sl@0
   343
	//Construct the resource file name buffer
sl@0
   344
	TRscEntry currentEntry=iRscList[iCurrentIndex];
sl@0
   345
	HBufC* rscFileName=HBufC::NewLC(currentEntry.iRscName.Length());
sl@0
   346
	TPtr modifiable(rscFileName->Des());
sl@0
   347
	modifiable.Copy(currentEntry.iRscName);
sl@0
   348
	//Construct the CResourceFile object	
sl@0
   349
	CResourceFile* resourceFile=CResourceFile::NewL(currentEntry.iRscData);
sl@0
   350
	CleanupStack::Pop(rscFileName);
sl@0
   351
	//update the index
sl@0
   352
	iCurrentIndex++;
sl@0
   353
	aRscFileName=rscFileName;
sl@0
   354
	return resourceFile;
sl@0
   355
	}
sl@0
   356
sl@0
   357
/** Reset the archive reader to start reading from the first resource
sl@0
   358
@internalComponent 
sl@0
   359
*/
sl@0
   360
void CResourceArchiveImpl::Reset()
sl@0
   361
	{
sl@0
   362
	iCurrentIndex=0;
sl@0
   363
	}
sl@0
   364
sl@0
   365
/** Return the type of the resource archive(SPI) file being read
sl@0
   366
@internalComponent 
sl@0
   367
*/
sl@0
   368
TUid CResourceArchiveImpl::Type()
sl@0
   369
	{
sl@0
   370
	return iSpiFileType;
sl@0
   371
	}
sl@0
   372
sl@0
   373
/** Function to look ahead whether next resource exists in the resource archive
sl@0
   374
@internalComponent
sl@0
   375
*/
sl@0
   376
TBool CResourceArchiveImpl::NextResourceExist() const
sl@0
   377
	{
sl@0
   378
	return (iCurrentIndex<iRscList.Count());
sl@0
   379
	}
sl@0
   380
sl@0
   381
/** Function used to retrieve a 4 bytes signed integer stored in Little Endian Format
sl@0
   382
@internalComponent
sl@0
   383
*/
sl@0
   384
TInt32 CResourceArchiveImpl::LittleEndianFourByteInteger(const TDesC8& aBuffer,TInt aIndexOfFirstByte) const
sl@0
   385
	{
sl@0
   386
	__ASSERT_DEBUG((aIndexOfFirstByte + 3) < aBuffer.Length(), ::Panic(EBafPanicBadIndex));
sl@0
   387
	return aBuffer[aIndexOfFirstByte] | (aBuffer[aIndexOfFirstByte+1]<<8) | (aBuffer[aIndexOfFirstByte+2]<<16) | (aBuffer[aIndexOfFirstByte+3]<<24);
sl@0
   388
	}
sl@0
   389
sl@0
   390
sl@0
   391
sl@0
   392
sl@0
   393
sl@0
   394
sl@0
   395
sl@0
   396
sl@0
   397
sl@0
   398
sl@0
   399
sl@0
   400
sl@0
   401
sl@0
   402
sl@0
   403
sl@0
   404
sl@0
   405
sl@0
   406
sl@0
   407
sl@0
   408
sl@0
   409