1.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000
1.2 +++ b/os/persistentdata/persistentstorage/dbms/security/SC_TextIn.cpp Fri Jun 15 03:10:57 2012 +0200
1.3 @@ -0,0 +1,840 @@
1.4 +// Copyright (c) 2004-2009 Nokia Corporation and/or its subsidiary(-ies).
1.5 +// All rights reserved.
1.6 +// This component and the accompanying materials are made available
1.7 +// under the terms of "Eclipse Public License v1.0"
1.8 +// which accompanies this distribution, and is available
1.9 +// at the URL "http://www.eclipse.org/legal/epl-v10.html".
1.10 +//
1.11 +// Initial Contributors:
1.12 +// Nokia Corporation - initial contribution.
1.13 +//
1.14 +// Contributors:
1.15 +//
1.16 +// Description:
1.17 +// CPDTextLoader class
1.18 +//
1.19 +//
1.20 +
1.21 +#define __INCLUDE_CAPABILITY_NAMES__
1.22 +#define __REFERENCE_CAPABILITY_NAMES__
1.23 +#include "e32capability.h"
1.24 +
1.25 +#include "SC_Strings.h"
1.26 +#include "SC_TextIn.h"
1.27 +
1.28 +namespace DBSC
1.29 +{
1.30 +
1.31 +/**
1.32 +Max capability count, when SID or VID used.
1.33 +@internalComponent
1.34 +*/
1.35 +const TInt KMaxCapabilityCount1 = 3;
1.36 +
1.37 +/**
1.38 +Max capability count, when no SID and no VID are used.
1.39 +@internalComponent
1.40 +*/
1.41 +const TInt KMaxCapabilityCount2 = 7;
1.42 +
1.43 +static TInt CompareCapabilities(const TCapability& aLeft, const TCapability& aRight)
1.44 + {
1.45 + return aRight - aLeft;
1.46 + }
1.47 +
1.48 +/**
1.49 +TStmtProps describes an object representing text file statement type and value
1.50 +(the right side of "=" expression)
1.51 +@internalComponent
1.52 +*/
1.53 +struct TStmtProps
1.54 + {
1.55 + inline TStmtProps() :
1.56 + iType(EStmtTEof),
1.57 + iValue(NULL, 0)
1.58 + {
1.59 + }
1.60 + TStmtType iType;
1.61 + TPtrC iValue;
1.62 + };
1.63 +
1.64 +/**
1.65 +StmtType2Class() function returns the statement class of the supplied statement type parameter.
1.66 +@param aType Statement type.
1.67 +@return Statement class.
1.68 +@internalComponent
1.69 +*/
1.70 +static TStmtClass StmtType2Class(TStmtType aType)
1.71 + {
1.72 + switch(aType)
1.73 + {
1.74 + case EStmtTComment:
1.75 + case EStmtTBlank:
1.76 + return EStmtCNoData;
1.77 + case EStmtTDatabase:
1.78 + case EStmtTTable:
1.79 + return EStmtCPolicyObj;
1.80 + case EStmtTRead:
1.81 + case EStmtTWrite:
1.82 + case EStmtTSchema:
1.83 + return EStmtCPolicyType;
1.84 + case EStmtTCapability:
1.85 + case EStmtTSID:
1.86 + case EStmtTVID:
1.87 + return EStmtCPolicyItem;
1.88 + case EStmtTBackup:
1.89 + return EStmtCBackup;
1.90 + default:
1.91 + break;
1.92 + }
1.93 + return EStmtCInvalid;
1.94 + }
1.95 +
1.96 +/**
1.97 +Capabilities count.
1.98 +@internalComponent
1.99 +*/
1.100 +
1.101 +const TInt KCapabilityCount = sizeof(CapabilityNames) / sizeof(CapabilityNames[0]);
1.102 +
1.103 +/**
1.104 +CapabilityName2Id() function searches and returns the related capability ID having
1.105 +the capability name as an input parameter.
1.106 +@param aName Capability name
1.107 +@return Related to aName capability ID
1.108 +@internalComponent
1.109 +*/
1.110 +static TCapability CapabilityName2Id(const TDesC& aName)
1.111 + {
1.112 + const TInt KMaxCapabilityStringLen = 20;
1.113 +
1.114 + TBufC8<KMaxCapabilityStringLen> cap;
1.115 + TPtr8 capPtr (cap.Des());
1.116 +
1.117 + capPtr.Copy(aName);
1.118 +
1.119 + for(TInt i=0;i<KCapabilityCount;++i)
1.120 + {
1.121 + if(0 == capPtr.CompareF(TBuf8<20>((TText8*)CapabilityNames[i])))
1.122 + return (TCapability)i;
1.123 + }
1.124 + return (TCapability) -1; // Return 'None' if no other capabilities are found
1.125 + }
1.126 +
1.127 +/**
1.128 +Statement keywords, which format is:
1.129 +<Keyword>
1.130 +KStmtKeywordT1 array is in 1:1 relation with KStmtT1 array, except the last KStmtT1
1.131 +member - EStmtTInvalid - it does not have a match in KStmtKeywordT1 array.
1.132 +@internalComponent
1.133 +*/
1.134 +const TDesC* const KStmtKeywordT1[] =
1.135 + {
1.136 + &KDbStr(), &KTblStr(), &KReadStr(), &KWriteStr(), &KSchemaStr(), &KBackupStr()
1.137 + };
1.138 +
1.139 +/**
1.140 +Statements count, which format is:
1.141 +<Keyword>
1.142 +@internalComponent
1.143 +*/
1.144 +const TInt KStmtT1Count = sizeof(KStmtKeywordT1) / sizeof(KStmtKeywordT1[0]);
1.145 +
1.146 +/**
1.147 +Statement IDs, which format is:
1.148 +<Keyword>
1.149 +KStmtKeywordT1 array is in 1:1 relation with KStmtT1 array, except the last KStmtT1
1.150 +member - EStmtTInvalid - it does not have a match in KStmtKeywordT1 array.
1.151 +"EStmtTInvalid" always has to be the last array element.
1.152 +@internalComponent
1.153 +*/
1.154 +const TStmtType KStmtT1[KStmtT1Count + 1] =
1.155 + {
1.156 + EStmtTDatabase, EStmtTTable, EStmtTRead, EStmtTWrite, EStmtTSchema, EStmtTBackup, EStmtTInvalid
1.157 + };
1.158 +
1.159 +/**
1.160 +StmtKeywordT1ToId() function searches and returns the related statement ID having
1.161 +the statement keyword as an input parameter.
1.162 +@param aKeyword Statement keyword
1.163 +@return Related to aKeyword statement ID
1.164 +@internalComponent
1.165 +*/
1.166 +static TStmtType StmtKeywordT1ToId(const TDesC& aKeyword)
1.167 + {
1.168 + TInt i;
1.169 + for(i=0;i<KStmtT1Count && aKeyword.CompareF(*(KStmtKeywordT1[i]));++i)
1.170 + {
1.171 + }
1.172 + return KStmtT1[i];
1.173 + }
1.174 +
1.175 +/**
1.176 +Statement keywords, which format is:
1.177 +<Keyword>=<Data>
1.178 +KStmtKeywordT2 array is in 1:1 relation with KStmtT2 array, except the last KStmtT2
1.179 +member - EStmtTInvalid - it does not have a match in KStmtKeywordT2 array.
1.180 +@internalComponent
1.181 +*/
1.182 +const TDesC* const KStmtKeywordT2[] =
1.183 + {
1.184 + &KNameStr(), &KCapabilityStr(), &KSIDStr(), &KVIDStr()
1.185 + };
1.186 +
1.187 +/**
1.188 +Statements count, which format is:
1.189 +<Keyword>=<Data>
1.190 +@internalComponent
1.191 +*/
1.192 +const TInt KStmtT2Count = sizeof(KStmtKeywordT2) / sizeof(KStmtKeywordT2[0]);
1.193 +
1.194 +/**
1.195 +Statement IDs, which format is:
1.196 +<Keyword>=<Data>
1.197 +KStmtKeywordT2 array is in 1:1 relation with KStmtT2 array, except the last KStmtT2
1.198 +member - EStmtTInvalid - it does not have a match in KStmtKeywordT2 array.
1.199 +EStmtTInvalid always has to be the last element in KStmtT2 array.
1.200 +@internalComponent
1.201 +*/
1.202 +const TStmtType KStmtT2[KStmtT2Count + 1] =
1.203 + {
1.204 + EStmtTName, EStmtTCapability, EStmtTSID, EStmtTVID, EStmtTInvalid
1.205 + };
1.206 +
1.207 +/**
1.208 +StmtKeywordT2ToId() function searches and returns the related statement ID having
1.209 +the statement keyword as an input parameter.
1.210 +@param aKeyword Statement keyword
1.211 +@return Related to aKeyword statement ID
1.212 +@internalComponent
1.213 +*/
1.214 +static TStmtType StmtKeywordT2ToId(const TDesC& aKeyword)
1.215 + {
1.216 + TInt i;
1.217 + for(i=0;i<KStmtT2Count && aKeyword.CompareF(*(KStmtKeywordT2[i]));++i)
1.218 + {
1.219 + }
1.220 + return KStmtT2[i];
1.221 + }
1.222 +
1.223 +/**
1.224 +TDes::Trim() does not work properly when the descriptor length is 0,
1.225 +or the descriptor holds a NULL pointer.
1.226 +@internalComponent
1.227 +*/
1.228 +static void Trim(TDes& aDes)
1.229 + {
1.230 + if(aDes.Length() > 0)
1.231 + {
1.232 + aDes.Trim();
1.233 + }
1.234 + }
1.235 +
1.236 +/**
1.237 +StmtType2PolicyType() function returns the related to aStmtType value - policy type.
1.238 +@param aStmtType Statement type
1.239 +@return The related policy type - R/W/S
1.240 +@internalComponent
1.241 +*/
1.242 +static TPolicyType StmtType2PolicyType(TStmtType aStmtType)
1.243 + {
1.244 + switch(aStmtType)
1.245 + {
1.246 + case EStmtTRead:
1.247 + return EPTRead;
1.248 + case EStmtTWrite:
1.249 + return EPTWrite;
1.250 + case EStmtTSchema:
1.251 + return EPTSchema;
1.252 + default:
1.253 + break;
1.254 + }
1.255 + return EPTNone;
1.256 + }
1.257 +
1.258 +/**
1.259 +Creates TSecurityPolicy instance of type 1: SID and a set of up to 3 capabilities.
1.260 +@param aSid Security ID
1.261 +@param aCapabilities Capabilities array.
1.262 +@param aSecurityPolicy Output. Created security policy.
1.263 +@internalComponent
1.264 +*/
1.265 +static void CreateSecurityPolicyT1(TSecureId aSid, const RArray<TCapability>& aCapabilities,
1.266 + TSecurityPolicy& aSecurityPolicy)
1.267 + {
1.268 + TInt count = aCapabilities.Count();
1.269 + if(count == 0)
1.270 + {
1.271 + aSecurityPolicy = TSecurityPolicy(aSid);
1.272 + }
1.273 + else if(count == 1)
1.274 + {
1.275 + aSecurityPolicy = TSecurityPolicy(aSid, aCapabilities[0]);
1.276 + }
1.277 + else if(count == 2)
1.278 + {
1.279 + aSecurityPolicy = TSecurityPolicy(aSid, aCapabilities[0], aCapabilities[1]);
1.280 + }
1.281 + else if(count == 3)
1.282 + {
1.283 + aSecurityPolicy = TSecurityPolicy(aSid, aCapabilities[0], aCapabilities[1], aCapabilities[2]);
1.284 + }
1.285 + else
1.286 + {
1.287 + User::Invariant();
1.288 + }
1.289 + }
1.290 +
1.291 +/**
1.292 +Creates TSecurityPolicy instance of type 2: VID and a set of up to 3 capabilities.
1.293 +@param aVid Vendor ID
1.294 +@param aCapabilities Capabilities array.
1.295 +@param aSecurityPolicy Output. Created security policy.
1.296 +@internalComponent
1.297 +*/
1.298 +static void CreateSecurityPolicyT2(TVendorId aVid, const RArray<TCapability>& aCapabilities,
1.299 + TSecurityPolicy& aSecurityPolicy)
1.300 + {
1.301 + TInt count = aCapabilities.Count();
1.302 + if(count == 0)
1.303 + {
1.304 + aSecurityPolicy = TSecurityPolicy(aVid);
1.305 + }
1.306 + else if(count == 1)
1.307 + {
1.308 + aSecurityPolicy = TSecurityPolicy(aVid, aCapabilities[0]);
1.309 + }
1.310 + else if(count == 2)
1.311 + {
1.312 + aSecurityPolicy = TSecurityPolicy(aVid, aCapabilities[0], aCapabilities[1]);
1.313 + }
1.314 + else if(count == 3)
1.315 + {
1.316 + aSecurityPolicy = TSecurityPolicy(aVid, aCapabilities[0], aCapabilities[1], aCapabilities[2]);
1.317 + }
1.318 + else
1.319 + {
1.320 + User::Invariant();
1.321 + }
1.322 + }
1.323 +
1.324 +/**
1.325 +Creates TSecurityPolicy instance of type 3: A set of up to 7 capabilities.
1.326 +@param aCapabilities Capabilities array.
1.327 +@param aSecurityPolicy Output. Created security policy.
1.328 +@internalComponent
1.329 +*/
1.330 +static void CreateSecurityPolicyT3(const RArray<TCapability>& aCapabilities, TSecurityPolicy& aSecurityPolicy)
1.331 + {
1.332 + TInt count = aCapabilities.Count();
1.333 + if(count == 1)
1.334 + {
1.335 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0]);
1.336 + }
1.337 + else if(count == 2)
1.338 + {
1.339 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1]);
1.340 + }
1.341 + else if(count == 3)
1.342 + {
1.343 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1], aCapabilities[2]);
1.344 + }
1.345 + else if(count == 4)
1.346 + {
1.347 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1], aCapabilities[2], aCapabilities[3]);
1.348 + }
1.349 + else if(count == 5)
1.350 + {
1.351 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1], aCapabilities[2], aCapabilities[3], aCapabilities[4]);
1.352 + }
1.353 + else if(count == 6)
1.354 + {
1.355 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1], aCapabilities[2], aCapabilities[3], aCapabilities[4], aCapabilities[5]);
1.356 + }
1.357 + else if(count == 7)
1.358 + {
1.359 + aSecurityPolicy = TSecurityPolicy(aCapabilities[0], aCapabilities[1], aCapabilities[2], aCapabilities[3], aCapabilities[4], aCapabilities[5], aCapabilities[6]);
1.360 + }
1.361 + else
1.362 + {
1.363 + User::Invariant();
1.364 + }
1.365 + }
1.366 +
1.367 +/**
1.368 +Creates TSecurityPolicy instance (initializing aSecurityPolicy parameter).
1.369 +@param aSid Security ID
1.370 +@param aVid Vendor ID
1.371 +@param aCapabilities Capabilities array.
1.372 +@leave KErrCorrupt Bad set of SID/VID/Capabilities
1.373 +@internalComponent
1.374 +*/
1.375 +static void CreateSecurityPolicyL(TSecureId aSid, TVendorId aVid,
1.376 + const RArray<TCapability>& aCapabilities,
1.377 + TSecurityPolicy& aSecurityPolicy)
1.378 + {
1.379 + TInt cababilityCount = aCapabilities.Count();
1.380 + if(aSid != 0 && aVid != 0)
1.381 + {
1.382 + __LEAVE(KErrCorrupt);
1.383 + }
1.384 + if(aSid != 0 || aVid != 0)
1.385 + {
1.386 + if(cababilityCount > KMaxCapabilityCount1)
1.387 + {
1.388 + __LEAVE(KErrCorrupt);
1.389 + }
1.390 + if(aSid != 0)
1.391 + {
1.392 + DBSC::CreateSecurityPolicyT1(aSid, aCapabilities, aSecurityPolicy);
1.393 + }
1.394 + else
1.395 + {
1.396 + DBSC::CreateSecurityPolicyT2(aVid, aCapabilities, aSecurityPolicy);
1.397 + }
1.398 + }
1.399 + else if(cababilityCount > KMaxCapabilityCount2 || cababilityCount == 0)
1.400 + {
1.401 + __LEAVE(KErrCorrupt);
1.402 + }
1.403 + else
1.404 + {
1.405 + DBSC::CreateSecurityPolicyT3(aCapabilities, aSecurityPolicy);
1.406 + }
1.407 + }
1.408 +
1.409 +/**
1.410 +*/
1.411 +inline CPDTextLoader::CPDTextLoader()
1.412 + {
1.413 + }
1.414 +
1.415 +/**
1.416 +Standard phase-one factory method for CPDTextLoader instance.
1.417 +CPDTextLoader instance will be used for loading a set of security policies information
1.418 +from a text file, creating the related security policy objects and adding them to
1.419 +a CPolicyDomain collection.
1.420 +@param aFs File server session
1.421 +@param aTextFileName Full text file path, which will be used as an input.
1.422 +@return A pointer to just created CPDTextLoader instance.
1.423 +@leave System-wide error codes, including KErrNoMemory.
1.424 +*/
1.425 +CPDTextLoader* CPDTextLoader::NewLC(RFs& aFs, const TDesC& aTextFileName)
1.426 + {
1.427 + CPDTextLoader* self = new (ELeave) CPDTextLoader;
1.428 + CleanupStack::PushL(self);
1.429 + self->ConstructL(aFs, aTextFileName);
1.430 + return self;
1.431 + }
1.432 +
1.433 +/**
1.434 +*/
1.435 +CPDTextLoader::~CPDTextLoader()
1.436 + {
1.437 + iRdStream.Close();
1.438 + }
1.439 +
1.440 +/**
1.441 +Standard phase-two construction method for CPDTextLoader instance.
1.442 +@param aFs File server session
1.443 +@param aTextFileName Full text file path, which will be used as an input.
1.444 +*/
1.445 +void CPDTextLoader::ConstructL(RFs& aFs, const TDesC& aTextFileName)
1.446 + {
1.447 + __LEAVE_IF_ERROR(iRdStream.Open(aFs, aTextFileName, EFileRead | EFileStreamText));
1.448 + }
1.449 +
1.450 +/**
1.451 +MPolicyDomainLoader::RunL() implementation, which is used to load security policies
1.452 +from a text file, create the related security policy objects and add them
1.453 +to CPolicyDomain instance, controlled by aPolicyDomainBuilder object.
1.454 +It is not called directly, but will be called back.
1.455 +@param aPolicyDomainBuilder TPolicyDomainBuilder instance, which will be used to add
1.456 + created security policy objects to the controlled by it collection.
1.457 +@leave System-wide error codes
1.458 +*/
1.459 +void CPDTextLoader::RunL(TPolicyDomainBuilder& aPolicyDomainBuilder)
1.460 + {
1.461 + TStmtProps stmtProps;
1.462 + const CDbPolicy* dbPolicy = LoadDbPolicyL(aPolicyDomainBuilder, stmtProps);
1.463 + __ASSERT(dbPolicy);
1.464 + LoadTblPoliciesL(aPolicyDomainBuilder, stmtProps, dbPolicy);
1.465 + LoadBackupSIDL(aPolicyDomainBuilder, stmtProps);
1.466 + }
1.467 +
1.468 +/**
1.469 +The method returns ETrue if this is the end of file.
1.470 +@return ETrue - EOF, EFalse otherwise
1.471 +*/
1.472 +TBool CPDTextLoader::IsEofL()
1.473 + {
1.474 + return iRdStream.Source()->TellL(MStreamBuf::ERead) >= iRdStream.Source()->SizeL();
1.475 + }
1.476 +
1.477 +/**
1.478 +The method parses a line from the text file, detects its type and gets the right side
1.479 +of "=" as a text line data descriptor. The information will be stored in aStmtProps
1.480 +parameter. If the text line is not recognizable, the method leaves with KErrCorrupt.
1.481 +Recognizable text line formats are:
1.482 +<[keyword]>
1.483 +<keyword>
1.484 +<keyword>=<value>
1.485 +<;>[comments]
1.486 +<blank line>
1.487 +@param aStmt Current text line
1.488 +@param aStmtProps The collected information will be stored there. Output parameter.
1.489 +@leave KErrCorrupt - the text line has unknown format
1.490 +*/
1.491 +void CPDTextLoader::GetStmtPropsL(const TDesC& aStmt, TStmtProps& aStmtProps) const
1.492 + {
1.493 + aStmtProps.iValue.Set(aStmt);
1.494 + if(aStmt.Length() == 0)
1.495 + {
1.496 + aStmtProps.iType = EStmtTBlank;
1.497 + return;
1.498 + }
1.499 + else if(aStmt[0] == ';')
1.500 + {
1.501 + aStmtProps.iType = EStmtTComment;
1.502 + return;
1.503 + }
1.504 + TBool res = TryGetStmt1Props(aStmt, aStmtProps);
1.505 + if(!res)
1.506 + {
1.507 + res = TryGetStmt2Props(aStmt, aStmtProps);
1.508 + }
1.509 + if(!res)
1.510 + {
1.511 + __LEAVE(KErrCorrupt);
1.512 + }
1.513 + }
1.514 +
1.515 +/**
1.516 +Tries to process a text line as a:
1.517 +<keyword>
1.518 +or
1.519 +<[keyword]>
1.520 +@param aStmt Current text line
1.521 +@param aStmtProps Output parameter
1.522 +@return ETrue, if it is recognizable text line. Then the method will set the text line type
1.523 + in aStmtProps.iType data member.
1.524 + EFalse This is not recognizable text line with <keyword> or <[keyword]> format.
1.525 +*/
1.526 +TBool CPDTextLoader::TryGetStmt1Props(const TDesC& aStmt, TStmtProps& aStmtProps) const
1.527 + {
1.528 + aStmtProps.iType = DBSC::StmtKeywordT1ToId(aStmt);
1.529 + return aStmtProps.iType != EStmtTInvalid;
1.530 + }
1.531 +
1.532 +/**
1.533 +Tries to process a text line as a:
1.534 +<keyword>=<value>
1.535 +@param aStmt Current text line
1.536 +@param aStmtProps Output parameter
1.537 +@return ETrue, if it is recognizable text line. Then the method will set the text line type
1.538 + in aStmtProps.iType data member and the line value in aStmtProps.iValue
1.539 + data member. The text will be converted to a upper case.
1.540 + EFalse This is not recognizable text line with <keyword>=<value> format.
1.541 +*/
1.542 +TBool CPDTextLoader::TryGetStmt2Props(const TDesC& aStmt, TStmtProps& aStmtProps) const
1.543 + {
1.544 + TInt eqPos = aStmt.Find(KEqStr);
1.545 + if(eqPos != KErrNotFound && eqPos < (aStmt.Length() - 1))
1.546 + {
1.547 + TPtr stmtKeyword(const_cast <TText*> (aStmt.Left(eqPos).Ptr()), eqPos, eqPos);
1.548 + DBSC::Trim(stmtKeyword);
1.549 + aStmtProps.iType = DBSC::StmtKeywordT2ToId(stmtKeyword);
1.550 + if(aStmtProps.iType != EStmtTInvalid)
1.551 + {
1.552 + TInt valPos = eqPos + 1;
1.553 + TInt valLen = aStmt.Length() - valPos;
1.554 + TPtr value(const_cast <TText*> (aStmt.Mid(valPos).Ptr()), valLen, valLen);
1.555 + DBSC::Trim(value);
1.556 + aStmtProps.iValue.Set(value);
1.557 + return ETrue;
1.558 + }
1.559 + }
1.560 + return EFalse;
1.561 + }
1.562 +
1.563 +/**
1.564 +The method loads a single text line from the file in the place, pointed by aStmt parameter.
1.565 +@param aStmt The place, where the text line data will be stored
1.566 +@return ETrue This not the end of file, the information in aStmt is valid.
1.567 + EFalse End of file.
1.568 +@leave System-wide error codes, including KErrCorrupt - unknown file format.
1.569 + */
1.570 +TBool CPDTextLoader::LoadStmtL(TPtr& aStmt)
1.571 + {
1.572 + if(IsEofL())
1.573 + {
1.574 + return EFalse;
1.575 + }
1.576 + TChar char_LF('\n');
1.577 + iStmt8.Zero();
1.578 + iRdStream.ReadL(iStmt8, char_LF);
1.579 + aStmt.Copy(iStmt8);
1.580 + const TInt len = aStmt.Length();
1.581 + if(len < 2)
1.582 + {//Unknown text file format. The text line should have at the end CR-LF pair.
1.583 + __LEAVE(KErrCorrupt);
1.584 + }
1.585 + if(TChar(aStmt[len - 1]) != char_LF)
1.586 + {//Too long line
1.587 + __LEAVE(KErrCorrupt);
1.588 + }
1.589 + aStmt.SetLength(len - 1);
1.590 + //The last character is (CR). Check for (LF).
1.591 + TChar char_CR('\r');
1.592 + if(TChar(aStmt[len - 2]) == char_CR)
1.593 + {
1.594 + aStmt.SetLength(len - 2);
1.595 + }
1.596 + DBSC::Trim(aStmt);
1.597 + return ETrue;
1.598 + }
1.599 +
1.600 +/**
1.601 +The method loads a single text line from the file in the place, pointed by iStmt data member
1.602 +skipping lines with comments and blank lines.
1.603 +@param aStmtProps Output parameter. It will be initialized after the call.
1.604 +@return Statement class.
1.605 +@leave System-wide error codes, including KErrCorrupt - unknown file format.
1.606 +*/
1.607 +TStmtClass CPDTextLoader::LoadNextStmtL(TStmtProps& aStmtProps)
1.608 + {
1.609 + TPtr stmt(const_cast <TText*> (iStmt.Ptr()), 0, iStmt.MaxLength());
1.610 + TStmtClass stmtClass = EStmtCInvalid;
1.611 + do
1.612 + {
1.613 + if(!LoadStmtL(stmt))
1.614 + {
1.615 + aStmtProps.iType = EStmtTEof;
1.616 + break;
1.617 + }
1.618 + GetStmtPropsL(stmt, aStmtProps);
1.619 + }
1.620 + while((stmtClass = DBSC::StmtType2Class(aStmtProps.iType)) == EStmtCNoData);
1.621 + return stmtClass;
1.622 + }
1.623 +
1.624 +/**
1.625 +The method loads a single text line from the file in the place, pointed by iStmt data member
1.626 +skipping lines with comments and blank lines. The loaded text line type is expected to be
1.627 +aStmtType.
1.628 +@param aStmtProps Output parameter. It will be initialized after the call.
1.629 +@param aStmtType Expected type of the loaded text line.
1.630 +@leave System-wide error codes, including KErrCorrupt - unknown file format or the loaded line type is
1.631 + not the expected type.
1.632 +*/
1.633 +void CPDTextLoader::LoadNextStmtOfTypeL(TStmtProps& aStmtProps, TStmtType aStmtType)
1.634 + {
1.635 + (void)LoadNextStmtL(aStmtProps);
1.636 + if(aStmtProps.iType != aStmtType)
1.637 + {
1.638 + __LEAVE(KErrCorrupt);
1.639 + }
1.640 + }
1.641 +
1.642 +/**
1.643 +The method loads all database policy related data from the text file.
1.644 +@param aPolicyDomainBuilder TPolicyDomainBuilder instance, which will be used to add
1.645 + created database security policy object to the controlled by it policy collection.
1.646 +@param aStmtProps The information about the last loaded text line.
1.647 +@return A const pointer to just created database policy object from loaded text data.
1.648 +@leave System-wide error codes.
1.649 +*/
1.650 +const CDbPolicy* CPDTextLoader::LoadDbPolicyL(TPolicyDomainBuilder& aPolicyDomainBuilder,
1.651 + TStmtProps& aStmtProps)
1.652 + {
1.653 + LoadNextStmtOfTypeL(aStmtProps, EStmtTDatabase);
1.654 +
1.655 + CPolicyBase::RPolicyCollection policyColl;
1.656 + CleanupClosePushL(policyColl);
1.657 +
1.658 + LoadSecurityPoliciesL(policyColl, aStmtProps);
1.659 +
1.660 + CDbPolicy* dbPolicy = CDbPolicy::NewLC(policyColl);
1.661 + aPolicyDomainBuilder.SetDbPolicyL(dbPolicy);
1.662 + CleanupStack::Pop(dbPolicy);
1.663 +
1.664 + CleanupStack::PopAndDestroy(&policyColl);
1.665 + return dbPolicy;
1.666 + }
1.667 +
1.668 +/**
1.669 +The method loads all table policy related data from the text file.
1.670 +@param aPolicyDomainBuilder TPolicyDomainBuilder instance, which will be used to add
1.671 + created table security policy objects to the controlled by it policy collection.
1.672 +@param aStmtProps The information about the last loaded text line.
1.673 +@param aDbPolicy A const pointer to the database policy object, created previously from loaded text data.
1.674 +@leave System-wide error codes.
1.675 +*/
1.676 +void CPDTextLoader::LoadTblPoliciesL(TPolicyDomainBuilder& aPolicyDomainBuilder,
1.677 + TStmtProps& aStmtProps, const CDbPolicy* aDbPolicy)
1.678 + {
1.679 + __ASSERT(aDbPolicy);
1.680 + CPolicyBase::RPolicyCollection policyColl;
1.681 + CleanupClosePushL(policyColl);
1.682 + while(aStmtProps.iType == EStmtTTable)
1.683 + {
1.684 + LoadNextStmtOfTypeL(aStmtProps, EStmtTName);
1.685 + TBufC<KMaxFileName> tableName;
1.686 + tableName = aStmtProps.iValue;
1.687 +
1.688 + LoadSecurityPoliciesL(policyColl, aStmtProps);
1.689 +
1.690 + CTblPolicy* tblPolicy = CTblPolicy::NewLC(tableName, policyColl, aDbPolicy);
1.691 + aPolicyDomainBuilder.AddTblPolicyL(tblPolicy);
1.692 + CleanupStack::Pop(tblPolicy);
1.693 + }
1.694 + CleanupStack::PopAndDestroy(&policyColl);
1.695 + }
1.696 +
1.697 +/**
1.698 +The method loads the backup & restore SID, if it is in the file.
1.699 +@param aPolicyDomainBuilder TPolicyDomainBuilder instance, which will be used to store
1.700 + loaded backup & restore SID.
1.701 +@param aStmtProps The information about the last loaded text line.
1.702 +@leave System-wide error codes.
1.703 +*/
1.704 +void CPDTextLoader::LoadBackupSIDL(TPolicyDomainBuilder& aPolicyDomainBuilder,
1.705 + TStmtProps& aStmtProps)
1.706 + {
1.707 + TSecureId backupSID((TUint32)ECapability_None);//ECapability_None is used in TSecurityPolicy constructors too.
1.708 + if(aStmtProps.iType == EStmtTBackup)
1.709 + {
1.710 + LoadNextStmtOfTypeL(aStmtProps, EStmtTSID);
1.711 + backupSID = GetIdL(aStmtProps.iValue);
1.712 + }
1.713 + else if(aStmtProps.iType != EStmtTEof)
1.714 + {
1.715 + __LEAVE(KErrCorrupt);
1.716 + }
1.717 + aPolicyDomainBuilder.SetBackupSID(backupSID);
1.718 + }
1.719 +
1.720 +/**
1.721 +The method loads all database/table related security policy information from the text file.
1.722 +@param aPolicyColl Output parameter - an array, which elements type is CPolicyBase::RPolicyCollection.
1.723 + The collected from the text file security policy information wil be stored there.
1.724 +@param aStmtProps The information about the last loaded text line.
1.725 +@leave System-wide error codes.
1.726 +*/
1.727 +void CPDTextLoader::LoadSecurityPoliciesL(CPolicyBase::RPolicyCollection& aPolicyColl,
1.728 + TStmtProps& aStmtProps)
1.729 + {
1.730 + aPolicyColl.Reset();
1.731 + (void)LoadNextStmtL(aStmtProps);
1.732 + while(DBSC::StmtType2Class(aStmtProps.iType) == EStmtCPolicyType)
1.733 + {
1.734 + CPolicyBase::TPolicy policy;
1.735 + policy.iType = DBSC::StmtType2PolicyType(aStmtProps.iType);
1.736 + __ASSERT(policy.iType != EPTNone);
1.737 + LoadSecurityPolicyL(policy.iData, aStmtProps);
1.738 + __LEAVE_IF_ERROR(aPolicyColl.Append(policy));
1.739 + }
1.740 + if(aPolicyColl.Count() == 0)
1.741 + {
1.742 + __LEAVE(KErrCorrupt);
1.743 + }
1.744 + }
1.745 +
1.746 +/**
1.747 +The method loads a single security policy from the text file.
1.748 +@param aSecurityPolicy Output parameter - the information from the file will be stored there.
1.749 + The collected from the text file security policy information wil be stored there.
1.750 +@param aStmtProps The information about the last loaded text line.
1.751 +@leave System-wide error codes.
1.752 +*/
1.753 +void CPDTextLoader::LoadSecurityPolicyL(TSecurityPolicy& aSecurityPolicy,
1.754 + TStmtProps& aStmtProps)
1.755 + {
1.756 + TSecureId sid(0);
1.757 + TVendorId vid(0);
1.758 + RArray<TCapability> capabilities;
1.759 + CleanupClosePushL(capabilities);
1.760 + while(LoadNextStmtL(aStmtProps) == EStmtCPolicyItem)
1.761 + {
1.762 + if(aStmtProps.iType == EStmtTCapability)
1.763 + {
1.764 + GetCapabilitiesL(aStmtProps.iValue, capabilities);
1.765 + }
1.766 + else if(aStmtProps.iType == EStmtTSID)
1.767 + {
1.768 + if(sid != 0)
1.769 + {//duplicated SID text line
1.770 + __LEAVE(KErrCorrupt);
1.771 + }
1.772 + sid = GetIdL(aStmtProps.iValue);
1.773 + }
1.774 + else if(aStmtProps.iType == EStmtTVID)
1.775 + {
1.776 + if(vid != 0)
1.777 + {//duplicated VID text line
1.778 + __LEAVE(KErrCorrupt);
1.779 + }
1.780 + vid = GetIdL(aStmtProps.iValue);
1.781 + }
1.782 + else
1.783 + {
1.784 + __ASSERT(0);
1.785 + }
1.786 + }
1.787 + if(capabilities.Count() == 0 && sid == 0 && vid == 0)
1.788 + {//invalid security policy data
1.789 + __LEAVE(KErrCorrupt);
1.790 + }
1.791 + DBSC::CreateSecurityPolicyL(sid, vid, capabilities, aSecurityPolicy);
1.792 + CleanupStack::PopAndDestroy(&capabilities);
1.793 + }
1.794 +
1.795 +/**
1.796 +The method parses a string with capabilities information and puts found capabilities in aCapability
1.797 +output parameter.
1.798 +@param aCapabilityStr Capabilities string.
1.799 +@param aCapabilities The collected capabilities will be stored there.
1.800 +@leave System-wide error codes. KErrCorrupt, if aCapability is not 0, which means there are
1.801 + 2 or more capability strings for the same policy.
1.802 +*/
1.803 +void CPDTextLoader::GetCapabilitiesL(const TDesC& aCapabilityStr,
1.804 + RArray<TCapability>& aCapabilities) const
1.805 + {
1.806 + if(aCapabilities.Count() > 0)
1.807 + {//No more than one "capability" statement in the text file!
1.808 + __LEAVE(KErrCorrupt);
1.809 + }
1.810 + TLinearOrder<TCapability> linearOrder(&DBSC::CompareCapabilities);
1.811 + TLex lex(aCapabilityStr);
1.812 + for(TPtrC token=lex.NextToken();token.Length()!=0;token.Set(lex.NextToken()))
1.813 + {
1.814 + TCapability cap = DBSC::CapabilityName2Id(token);
1.815 + if(cap != ECapability_Limit)
1.816 + {//InsertInOrder() - to warn the user in case of duplicates
1.817 + __LEAVE_IF_ERROR(aCapabilities.InsertInOrder(cap, linearOrder));
1.818 + }
1.819 + else
1.820 + {
1.821 + __LEAVE(KErrGeneral);//Unknown capability
1.822 + }
1.823 + }
1.824 + }
1.825 +
1.826 +/**
1.827 +@param aStr A string with SID or VID.
1.828 +@return The UID, extracted from the string
1.829 +@leave System-wide error codes.
1.830 +*/
1.831 +TUint CPDTextLoader::GetIdL(const TDesC& aStr) const
1.832 + {
1.833 + TLex lex(aStr);
1.834 + TUint id;
1.835 + __LEAVE_IF_ERROR(lex.Val(id, EHex));
1.836 + if(id == 0)
1.837 + {
1.838 + __LEAVE(KErrCorrupt);
1.839 + }
1.840 + return id;
1.841 + }
1.842 +
1.843 +} //end of - namespace DBSC