os/persistentdata/featuremgmt/featureregistry/src/setup/featregsetup.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
// Copyright (c) 2005-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
// Reads feature config file, checks it is valid, defines feature property from it and quits.
sl@0
    15
// Debug builds panic if config file could not be read or is invalid, but only after all
sl@0
    16
// processing is complete so debug and release executables still work roughly the same.
sl@0
    17
// 
sl@0
    18
//
sl@0
    19
sl@0
    20
/**
sl@0
    21
 @file
sl@0
    22
 @internalComponent
sl@0
    23
*/
sl@0
    24
sl@0
    25
#include <e32base.h>
sl@0
    26
#include <e32cmn.h>
sl@0
    27
#include <e32property.h>
sl@0
    28
#include <f32file.h>
sl@0
    29
#include "featregpan.h"
sl@0
    30
#include "featregcmn.h"
sl@0
    31
sl@0
    32
static TBool IsFeatureDataValid(const TDesC8& aFeatureDes)
sl@0
    33
	{
sl@0
    34
	// size from number of entries and ranges in header must match descriptor size
sl@0
    35
	const TFeatureHeader* header = reinterpret_cast<const TFeatureHeader*>(aFeatureDes.Ptr());
sl@0
    36
	if (header->IsInvalidOrBadSize(aFeatureDes.Size()))
sl@0
    37
		{
sl@0
    38
		return EFalse;
sl@0
    39
		}
sl@0
    40
sl@0
    41
	// check feature entries are non-repeating and sorted from lowest to highest UID
sl@0
    42
	const TFeatureEntry* entry = reinterpret_cast<const TFeatureEntry*>(header + 1);
sl@0
    43
	TInt i = header->iFeatureEntryCount;
sl@0
    44
	TUint32 previousEntryUid = 0;
sl@0
    45
	if (i > 0)
sl@0
    46
		{
sl@0
    47
		previousEntryUid = entry->iUid;
sl@0
    48
		++entry;
sl@0
    49
		--i;
sl@0
    50
		}
sl@0
    51
	for (; i > 0; --i)
sl@0
    52
		{
sl@0
    53
		if (previousEntryUid >= entry->iUid)
sl@0
    54
			{
sl@0
    55
			return EFalse;
sl@0
    56
			}
sl@0
    57
		previousEntryUid = entry->iUid;
sl@0
    58
		++entry;
sl@0
    59
		}
sl@0
    60
sl@0
    61
	// check default=supported ranges are in low-high UID pairs
sl@0
    62
	const TFeatureRange* range = reinterpret_cast<const TFeatureRange*>(entry);
sl@0
    63
	for (i = header->iFeatureRangeCount; i > 0; --i)
sl@0
    64
		{
sl@0
    65
		if (range->iLowUid > range->iHighUid)
sl@0
    66
			{
sl@0
    67
			return EFalse;
sl@0
    68
			}
sl@0
    69
		++range;
sl@0
    70
		}
sl@0
    71
sl@0
    72
	return ETrue;
sl@0
    73
	}
sl@0
    74
sl@0
    75
/**
sl@0
    76
 * Reads contents of a file into aBuf, reallocating if needed
sl@0
    77
 * Afterwards aBuf may own memory even in case of failure - up to calling code to clean it up
sl@0
    78
 * @return KErrNone on success, KErrCorrupt if invalid file size, otherwise other system-wide error code
sl@0
    79
 */
sl@0
    80
static TInt ReadFeatureFileToBuf(RFs &aFs, const TDesC& aFileName, RBuf8& aBuf)
sl@0
    81
	{
sl@0
    82
	RFile inFile;
sl@0
    83
	TInt result = inFile.Open(aFs, aFileName, EFileRead|EFileShareReadersOnly|EFileStream);
sl@0
    84
	if (result != KErrNone)
sl@0
    85
		{
sl@0
    86
		return result;
sl@0
    87
		}
sl@0
    88
	TInt fileSize;
sl@0
    89
	result = inFile.Size(fileSize);
sl@0
    90
	if ((fileSize < sizeof(TFeatureHeader)) || (fileSize > RProperty::KMaxLargePropertySize))
sl@0
    91
		{
sl@0
    92
		result = KErrCorrupt;
sl@0
    93
		}
sl@0
    94
	if (result == KErrNone)
sl@0
    95
		{
sl@0
    96
 		result = aBuf.ReAlloc(fileSize);
sl@0
    97
		}
sl@0
    98
	if (result == KErrNone)
sl@0
    99
		{
sl@0
   100
		result = inFile.Read(aBuf);
sl@0
   101
		}
sl@0
   102
	inFile.Close();
sl@0
   103
	if (result == KErrNone && !IsFeatureDataValid(aBuf))
sl@0
   104
		{
sl@0
   105
		result = KErrCorrupt;
sl@0
   106
		}	
sl@0
   107
	return result;
sl@0
   108
	}
sl@0
   109
sl@0
   110
static TInt CompareFeatureEntry(const TFeatureEntry& aEntry1,const TFeatureEntry& aEntry2)
sl@0
   111
	{
sl@0
   112
	return (aEntry1.iUid-aEntry2.iUid);
sl@0
   113
	}
sl@0
   114
	
sl@0
   115
static TInt CompareFeatureRange(const TFeatureRange& aEntry1,const TFeatureRange& aEntry2)
sl@0
   116
	{
sl@0
   117
	//the comparison when both default supported range are the same
sl@0
   118
	if (aEntry1.iLowUid==aEntry2.iLowUid && aEntry1.iHighUid==aEntry2.iHighUid)
sl@0
   119
		{
sl@0
   120
		return 0;		
sl@0
   121
		}
sl@0
   122
	return (aEntry1.iLowUid<aEntry2.iLowUid?-1:1);
sl@0
   123
	}	
sl@0
   124
sl@0
   125
static void ResetRBufArray(RArray<RBuf8>& aRBuf8Array)
sl@0
   126
	{
sl@0
   127
	TInt count=aRBuf8Array.Count();
sl@0
   128
	for (TInt i=0;i<count;i++)
sl@0
   129
		{
sl@0
   130
		aRBuf8Array[i].Close();
sl@0
   131
		}
sl@0
   132
	aRBuf8Array.Reset();
sl@0
   133
	aRBuf8Array.Close();
sl@0
   134
	}
sl@0
   135
sl@0
   136
/**
sl@0
   137
This function will try to find and open all the "featreg.cfg" files from the path specified
sl@0
   138
process them in the order of increasing rom image id before copying the aggregate content
sl@0
   139
into the buffer aBuf. Validation of each of the file is also performed inside this function.
sl@0
   140
@param aFs an opened file server session
sl@0
   141
@param aPath the path to look for the featreg files
sl@0
   142
@param aBuf the buffer to contain the aggregated file content
sl@0
   143
@return KErrNoMemory if no memory, KErrCorrupt if any of the featreg.cfg files it found in the path
sl@0
   144
		is corrupt.
sl@0
   145
*/
sl@0
   146
GLDEF_C TInt ReadMultipleFeatureFileToBuf(RFs& aFs,const TDesC& aPath,RBuf8& aBuf)
sl@0
   147
	{
sl@0
   148
	CDir* featregList=NULL;
sl@0
   149
	HBufC* matchPattern=HBufC::New(aPath.Length()+KFeatregMatchPattern().Length());
sl@0
   150
	if (!matchPattern)
sl@0
   151
		{
sl@0
   152
		return KErrNoMemory;
sl@0
   153
		}
sl@0
   154
	TPtr matchPatternPtr(matchPattern->Des());		
sl@0
   155
	matchPatternPtr.Append(aPath);
sl@0
   156
	matchPatternPtr.Append(KFeatregMatchPattern);
sl@0
   157
	TInt ret=aFs.GetDir(*matchPattern,KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive,ESortByName,featregList);
sl@0
   158
	delete matchPattern;
sl@0
   159
	if (ret!=KErrNone)
sl@0
   160
		{
sl@0
   161
		return ret;
sl@0
   162
		}
sl@0
   163
	TInt fileCount=featregList->Count();
sl@0
   164
	//if only one file we do not need the array
sl@0
   165
	if (fileCount==1)
sl@0
   166
		{
sl@0
   167
		TFileName fullName(aPath);
sl@0
   168
		fullName.Append((*featregList)[0].iName);
sl@0
   169
		delete featregList;
sl@0
   170
		return ReadFeatureFileToBuf(aFs,fullName,aBuf);
sl@0
   171
		}
sl@0
   172
		
sl@0
   173
	//else if there are more than one file we need to open and validate each file
sl@0
   174
	//before appending each file content to the buffer array
sl@0
   175
	RArray<RBuf8> featregFileBufArray;
sl@0
   176
	for (TInt i=0;i<fileCount;i++)
sl@0
   177
		{
sl@0
   178
		TFileName fullName(aPath);
sl@0
   179
		fullName.Append((*featregList)[i].iName);
sl@0
   180
		RBuf8 fileBuffer;
sl@0
   181
		//first try to read the file into buffer and if successful append to the array
sl@0
   182
		if ((ret=ReadFeatureFileToBuf(aFs,fullName,fileBuffer))!=KErrNone || (ret=featregFileBufArray.Append(fileBuffer))!=KErrNone)
sl@0
   183
			{
sl@0
   184
			//something wrong here reset the buffer and return the error code
sl@0
   185
			fileBuffer.Close();
sl@0
   186
			delete featregList;
sl@0
   187
			ResetRBufArray(featregFileBufArray);
sl@0
   188
			return ret;
sl@0
   189
			}
sl@0
   190
		}
sl@0
   191
	//the list of files no longer needed we can delete now
sl@0
   192
	delete featregList;
sl@0
   193
sl@0
   194
	//at this stage we are sure all the featreg files are not corrupt
sl@0
   195
	//we need to read the single feature entries and the default
sl@0
   196
	//supported range before reconstructing the output buffer
sl@0
   197
	RArray<TFeatureEntry> singleArray;
sl@0
   198
	RArray<TFeatureRange> rangeArray;
sl@0
   199
	//process it in the reverse order as we read it as rom image with higher id will be mounted first
sl@0
   200
	for (TInt idx=fileCount-1;idx>=0;idx--)
sl@0
   201
		{
sl@0
   202
		const TFeatureHeader* header = reinterpret_cast<const TFeatureHeader*>(featregFileBufArray[idx].Ptr());
sl@0
   203
		const TFeatureEntry* entry=reinterpret_cast<const TFeatureEntry*>(header+1);
sl@0
   204
		const TFeatureRange* range=reinterpret_cast<const TFeatureRange*>(entry+header->iFeatureEntryCount);		
sl@0
   205
		for (TInt s=0;s<header->iFeatureEntryCount;s++)
sl@0
   206
			{
sl@0
   207
			ret=singleArray.InsertInOrder(*(reinterpret_cast<const TFeatureEntry*>(entry+s)),TLinearOrder<TFeatureEntry>(CompareFeatureEntry));
sl@0
   208
			//ignore KErrAlreadyExists as we purposely do not want duplicate entries
sl@0
   209
			if (ret!=KErrNone && ret!=KErrAlreadyExists)
sl@0
   210
				{
sl@0
   211
				singleArray.Close();
sl@0
   212
				rangeArray.Close();
sl@0
   213
				ResetRBufArray(featregFileBufArray);
sl@0
   214
				return ret;			
sl@0
   215
				}
sl@0
   216
			}
sl@0
   217
		//for range, we just insert them but ignore duplicate
sl@0
   218
		for (TInt r=0;r<header->iFeatureRangeCount;r++)
sl@0
   219
			{
sl@0
   220
			ret=rangeArray.InsertInOrder(*(reinterpret_cast<const TFeatureRange*>(range+r)),TLinearOrder<TFeatureRange>(CompareFeatureRange));
sl@0
   221
			if (ret!=KErrNone && ret!=KErrAlreadyExists)
sl@0
   222
				{
sl@0
   223
				singleArray.Close();
sl@0
   224
				rangeArray.Close();
sl@0
   225
				ResetRBufArray(featregFileBufArray);
sl@0
   226
				return ret;	
sl@0
   227
				}
sl@0
   228
			}
sl@0
   229
		}
sl@0
   230
	//now the final step is to construct the aggregate RBuf
sl@0
   231
	RBuf8 buf8;
sl@0
   232
	TInt singleCount=singleArray.Count();
sl@0
   233
	TInt rangeCount=rangeArray.Count();
sl@0
   234
	ret=buf8.Create(sizeof(TFeatureHeader)+singleCount*sizeof(TFeatureEntry)+rangeCount*sizeof(TFeatureRange));
sl@0
   235
	if (ret==KErrNone)
sl@0
   236
		{
sl@0
   237
		TFeatureHeader header={validTypePrefix,0,singleCount,rangeCount};
sl@0
   238
		buf8.Append(reinterpret_cast<const TUint8*>(&header),sizeof(TFeatureHeader));
sl@0
   239
		for (TInt s=0;s<singleCount;s++)
sl@0
   240
			{
sl@0
   241
			buf8.Append(reinterpret_cast<const TUint8*>(&(singleArray[s])),sizeof(TFeatureEntry));
sl@0
   242
			}
sl@0
   243
		for (TInt r=0;r<rangeCount;r++)
sl@0
   244
			{
sl@0
   245
			buf8.Append(reinterpret_cast<const TUint8*>(&(rangeArray[r])),sizeof(TFeatureRange));
sl@0
   246
			}	
sl@0
   247
		//transfer ownership now
sl@0
   248
		aBuf.Assign(buf8);	
sl@0
   249
		}
sl@0
   250
	//perform cleanup
sl@0
   251
	singleArray.Close();
sl@0
   252
	rangeArray.Close();
sl@0
   253
	ResetRBufArray(featregFileBufArray);	
sl@0
   254
	return ret;	
sl@0
   255
	}
sl@0
   256
sl@0
   257
sl@0
   258
/**
sl@0
   259
 * @return KErrNone (0) if successful, KErrNoMemory or KErrCorrupt if file read failed,
sl@0
   260
 * otherwise error result from calling RProperty::Define/Set
sl@0
   261
 */
sl@0
   262
#ifndef FEATREGSETUPTEST
sl@0
   263
TInt E32Main()
sl@0
   264
	{
sl@0
   265
	__UHEAP_MARK;
sl@0
   266
	CTrapCleanup* trapCleanup=CTrapCleanup::New();
sl@0
   267
	if (!trapCleanup)
sl@0
   268
		{
sl@0
   269
		return KErrNoMemory;
sl@0
   270
		}
sl@0
   271
sl@0
   272
	// if the following RProperty is defined, look for config file on the C: private
sl@0
   273
	// data cage instead of the normal Z:. Used for testing only.
sl@0
   274
	TInt testPropertyValue;
sl@0
   275
	TInt testResult = RProperty::Get(KFeaturePropCat, KFeatRegConfigTestKey, testPropertyValue);
sl@0
   276
	const TBool useCDataCage = testResult != KErrNotFound;
sl@0
   277
sl@0
   278
	RBuf8 featureBuf;
sl@0
   279
	RFs fs;
sl@0
   280
	TInt readResult = KErrNone;
sl@0
   281
	TInt publishResult = KErrNone;
sl@0
   282
	readResult = fs.Connect();
sl@0
   283
	if (readResult == KErrNone)
sl@0
   284
		{
sl@0
   285
		if (useCDataCage)
sl@0
   286
			{
sl@0
   287
			TConfigFileName filename;
sl@0
   288
			GetSystemDrivePath(filename);
sl@0
   289
			readResult = ReadFeatureFileToBuf(fs, filename, featureBuf);
sl@0
   290
			}
sl@0
   291
		else
sl@0
   292
			{
sl@0
   293
			readResult=ReadMultipleFeatureFileToBuf(fs,KFeatregRomPrivatePath,featureBuf);
sl@0
   294
			}
sl@0
   295
		fs.Close();
sl@0
   296
		}
sl@0
   297
sl@0
   298
	// lack of memory may be temporary, so end rather than publishing invalid property now
sl@0
   299
	if (readResult != KErrNoMemory)
sl@0
   300
		{
sl@0
   301
		RProcess thisProcess;
sl@0
   302
		// sanity check that feature property category in common header equals SID of this process
sl@0
   303
		ASSERT(KFeaturePropCat == thisProcess.SecureId());
sl@0
   304
		TSecurityPolicy readPolicy(TSecurityPolicy::EAlwaysPass);
sl@0
   305
		TSecurityPolicy writePolicy(thisProcess.SecureId());
sl@0
   306
		publishResult = RProperty::Define(KFeaturePropKey, RProperty::ELargeByteArray, readPolicy, writePolicy);
sl@0
   307
sl@0
   308
		if ((publishResult == KErrNone) || (publishResult == KErrAlreadyExists))
sl@0
   309
			{
sl@0
   310
			if (readResult == KErrNone)
sl@0
   311
				{
sl@0
   312
				publishResult = RProperty::Set(KFeaturePropCat, KFeaturePropKey, featureBuf);
sl@0
   313
				}
sl@0
   314
			else
sl@0
   315
				{
sl@0
   316
				// bad config file: set up invalid property with number of features/ranges not consistent
sl@0
   317
				// with size of property, so Feature Registry API always returns KErrCorrupt
sl@0
   318
				TFeatureHeader corruptHeader;
sl@0
   319
				corruptHeader.SetInvalid();
sl@0
   320
				TPckg<TFeatureHeader> corruptFeaturePckg(corruptHeader);
sl@0
   321
				publishResult = RProperty::Set(KFeaturePropCat, KFeaturePropKey, corruptFeaturePckg);
sl@0
   322
				}
sl@0
   323
			}
sl@0
   324
sl@0
   325
		if (readResult != KErrNone)
sl@0
   326
			{
sl@0
   327
			readResult = KErrCorrupt;
sl@0
   328
			// panic in debug mode alerts system integrators that config file is missing,
sl@0
   329
			// unreadable or invalid in this OS configuration: a serious problem
sl@0
   330
			__ASSERT_DEBUG(EFalse, Panic(EFeatRegBadConfig));
sl@0
   331
			}
sl@0
   332
		}
sl@0
   333
sl@0
   334
	featureBuf.Close();
sl@0
   335
	delete trapCleanup;
sl@0
   336
	__UHEAP_MARKEND;
sl@0
   337
	return (readResult != KErrNone) ? readResult : publishResult;
sl@0
   338
	}
sl@0
   339
#endif