diff -r 000000000000 -r bde4ae8d615e os/persistentdata/featuremgmt/featureregistry/src/setup/featregsetup.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/os/persistentdata/featuremgmt/featureregistry/src/setup/featregsetup.cpp Fri Jun 15 03:10:57 2012 +0200 @@ -0,0 +1,339 @@ +// Copyright (c) 2005-2009 Nokia Corporation and/or its subsidiary(-ies). +// All rights reserved. +// This component and the accompanying materials are made available +// under the terms of "Eclipse Public License v1.0" +// which accompanies this distribution, and is available +// at the URL "http://www.eclipse.org/legal/epl-v10.html". +// +// Initial Contributors: +// Nokia Corporation - initial contribution. +// +// Contributors: +// +// Description: +// Reads feature config file, checks it is valid, defines feature property from it and quits. +// Debug builds panic if config file could not be read or is invalid, but only after all +// processing is complete so debug and release executables still work roughly the same. +// +// + +/** + @file + @internalComponent +*/ + +#include +#include +#include +#include +#include "featregpan.h" +#include "featregcmn.h" + +static TBool IsFeatureDataValid(const TDesC8& aFeatureDes) + { + // size from number of entries and ranges in header must match descriptor size + const TFeatureHeader* header = reinterpret_cast(aFeatureDes.Ptr()); + if (header->IsInvalidOrBadSize(aFeatureDes.Size())) + { + return EFalse; + } + + // check feature entries are non-repeating and sorted from lowest to highest UID + const TFeatureEntry* entry = reinterpret_cast(header + 1); + TInt i = header->iFeatureEntryCount; + TUint32 previousEntryUid = 0; + if (i > 0) + { + previousEntryUid = entry->iUid; + ++entry; + --i; + } + for (; i > 0; --i) + { + if (previousEntryUid >= entry->iUid) + { + return EFalse; + } + previousEntryUid = entry->iUid; + ++entry; + } + + // check default=supported ranges are in low-high UID pairs + const TFeatureRange* range = reinterpret_cast(entry); + for (i = header->iFeatureRangeCount; i > 0; --i) + { + if (range->iLowUid > range->iHighUid) + { + return EFalse; + } + ++range; + } + + return ETrue; + } + +/** + * Reads contents of a file into aBuf, reallocating if needed + * Afterwards aBuf may own memory even in case of failure - up to calling code to clean it up + * @return KErrNone on success, KErrCorrupt if invalid file size, otherwise other system-wide error code + */ +static TInt ReadFeatureFileToBuf(RFs &aFs, const TDesC& aFileName, RBuf8& aBuf) + { + RFile inFile; + TInt result = inFile.Open(aFs, aFileName, EFileRead|EFileShareReadersOnly|EFileStream); + if (result != KErrNone) + { + return result; + } + TInt fileSize; + result = inFile.Size(fileSize); + if ((fileSize < sizeof(TFeatureHeader)) || (fileSize > RProperty::KMaxLargePropertySize)) + { + result = KErrCorrupt; + } + if (result == KErrNone) + { + result = aBuf.ReAlloc(fileSize); + } + if (result == KErrNone) + { + result = inFile.Read(aBuf); + } + inFile.Close(); + if (result == KErrNone && !IsFeatureDataValid(aBuf)) + { + result = KErrCorrupt; + } + return result; + } + +static TInt CompareFeatureEntry(const TFeatureEntry& aEntry1,const TFeatureEntry& aEntry2) + { + return (aEntry1.iUid-aEntry2.iUid); + } + +static TInt CompareFeatureRange(const TFeatureRange& aEntry1,const TFeatureRange& aEntry2) + { + //the comparison when both default supported range are the same + if (aEntry1.iLowUid==aEntry2.iLowUid && aEntry1.iHighUid==aEntry2.iHighUid) + { + return 0; + } + return (aEntry1.iLowUid& aRBuf8Array) + { + TInt count=aRBuf8Array.Count(); + for (TInt i=0;iDes()); + matchPatternPtr.Append(aPath); + matchPatternPtr.Append(KFeatregMatchPattern); + TInt ret=aFs.GetDir(*matchPattern,KEntryAttReadOnly | KEntryAttHidden | KEntryAttSystem | KEntryAttArchive,ESortByName,featregList); + delete matchPattern; + if (ret!=KErrNone) + { + return ret; + } + TInt fileCount=featregList->Count(); + //if only one file we do not need the array + if (fileCount==1) + { + TFileName fullName(aPath); + fullName.Append((*featregList)[0].iName); + delete featregList; + return ReadFeatureFileToBuf(aFs,fullName,aBuf); + } + + //else if there are more than one file we need to open and validate each file + //before appending each file content to the buffer array + RArray featregFileBufArray; + for (TInt i=0;i singleArray; + RArray rangeArray; + //process it in the reverse order as we read it as rom image with higher id will be mounted first + for (TInt idx=fileCount-1;idx>=0;idx--) + { + const TFeatureHeader* header = reinterpret_cast(featregFileBufArray[idx].Ptr()); + const TFeatureEntry* entry=reinterpret_cast(header+1); + const TFeatureRange* range=reinterpret_cast(entry+header->iFeatureEntryCount); + for (TInt s=0;siFeatureEntryCount;s++) + { + ret=singleArray.InsertInOrder(*(reinterpret_cast(entry+s)),TLinearOrder(CompareFeatureEntry)); + //ignore KErrAlreadyExists as we purposely do not want duplicate entries + if (ret!=KErrNone && ret!=KErrAlreadyExists) + { + singleArray.Close(); + rangeArray.Close(); + ResetRBufArray(featregFileBufArray); + return ret; + } + } + //for range, we just insert them but ignore duplicate + for (TInt r=0;riFeatureRangeCount;r++) + { + ret=rangeArray.InsertInOrder(*(reinterpret_cast(range+r)),TLinearOrder(CompareFeatureRange)); + if (ret!=KErrNone && ret!=KErrAlreadyExists) + { + singleArray.Close(); + rangeArray.Close(); + ResetRBufArray(featregFileBufArray); + return ret; + } + } + } + //now the final step is to construct the aggregate RBuf + RBuf8 buf8; + TInt singleCount=singleArray.Count(); + TInt rangeCount=rangeArray.Count(); + ret=buf8.Create(sizeof(TFeatureHeader)+singleCount*sizeof(TFeatureEntry)+rangeCount*sizeof(TFeatureRange)); + if (ret==KErrNone) + { + TFeatureHeader header={validTypePrefix,0,singleCount,rangeCount}; + buf8.Append(reinterpret_cast(&header),sizeof(TFeatureHeader)); + for (TInt s=0;s(&(singleArray[s])),sizeof(TFeatureEntry)); + } + for (TInt r=0;r(&(rangeArray[r])),sizeof(TFeatureRange)); + } + //transfer ownership now + aBuf.Assign(buf8); + } + //perform cleanup + singleArray.Close(); + rangeArray.Close(); + ResetRBufArray(featregFileBufArray); + return ret; + } + + +/** + * @return KErrNone (0) if successful, KErrNoMemory or KErrCorrupt if file read failed, + * otherwise error result from calling RProperty::Define/Set + */ +#ifndef FEATREGSETUPTEST +TInt E32Main() + { + __UHEAP_MARK; + CTrapCleanup* trapCleanup=CTrapCleanup::New(); + if (!trapCleanup) + { + return KErrNoMemory; + } + + // if the following RProperty is defined, look for config file on the C: private + // data cage instead of the normal Z:. Used for testing only. + TInt testPropertyValue; + TInt testResult = RProperty::Get(KFeaturePropCat, KFeatRegConfigTestKey, testPropertyValue); + const TBool useCDataCage = testResult != KErrNotFound; + + RBuf8 featureBuf; + RFs fs; + TInt readResult = KErrNone; + TInt publishResult = KErrNone; + readResult = fs.Connect(); + if (readResult == KErrNone) + { + if (useCDataCage) + { + TConfigFileName filename; + GetSystemDrivePath(filename); + readResult = ReadFeatureFileToBuf(fs, filename, featureBuf); + } + else + { + readResult=ReadMultipleFeatureFileToBuf(fs,KFeatregRomPrivatePath,featureBuf); + } + fs.Close(); + } + + // lack of memory may be temporary, so end rather than publishing invalid property now + if (readResult != KErrNoMemory) + { + RProcess thisProcess; + // sanity check that feature property category in common header equals SID of this process + ASSERT(KFeaturePropCat == thisProcess.SecureId()); + TSecurityPolicy readPolicy(TSecurityPolicy::EAlwaysPass); + TSecurityPolicy writePolicy(thisProcess.SecureId()); + publishResult = RProperty::Define(KFeaturePropKey, RProperty::ELargeByteArray, readPolicy, writePolicy); + + if ((publishResult == KErrNone) || (publishResult == KErrAlreadyExists)) + { + if (readResult == KErrNone) + { + publishResult = RProperty::Set(KFeaturePropCat, KFeaturePropKey, featureBuf); + } + else + { + // bad config file: set up invalid property with number of features/ranges not consistent + // with size of property, so Feature Registry API always returns KErrCorrupt + TFeatureHeader corruptHeader; + corruptHeader.SetInvalid(); + TPckg corruptFeaturePckg(corruptHeader); + publishResult = RProperty::Set(KFeaturePropCat, KFeaturePropKey, corruptFeaturePckg); + } + } + + if (readResult != KErrNone) + { + readResult = KErrCorrupt; + // panic in debug mode alerts system integrators that config file is missing, + // unreadable or invalid in this OS configuration: a serious problem + __ASSERT_DEBUG(EFalse, Panic(EFeatRegBadConfig)); + } + } + + featureBuf.Close(); + delete trapCleanup; + __UHEAP_MARKEND; + return (readResult != KErrNone) ? readResult : publishResult; + } +#endif