os/persistentdata/persistentstorage/sql/SRC/Client/SqlDatabaseImpl.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.
     1 // Copyright (c) 2005-2010 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 #include "SqlSecurityImpl.h"	//CSqlSecurityPolicy
    17 #include "SqlDatabaseImpl.h"	//CSqlDatabaseImpl
    18 #include <e32debug.h>
    19 
    20 /**
    21 Manual compaction - max compacton step execution time in microseconds.
    22 
    23 @internalComponent
    24 */
    25 static const TInt KCompactMaxStepTimeUs = 100000;
    26 
    27 /**
    28 Minimal amount of free database space to be removed by the compaction steps.
    29 
    30 @internalComponent
    31 */
    32 static const TInt KCompactMinStepSize =  2 * 1024;
    33 
    34 /**
    35 The amount of free database space to be removed by the first compaction step.
    36 
    37 @internalComponent
    38 */
    39 static const TInt KCompactStartStepSize = 32 * 1024;
    40 
    41 /**
    42 The aim of the function is to determine the maximum size of space to be freed, which fits within the time constraint.
    43 The decision is based on the time spent on the pervious compaction step.
    44 If the time is bigger than KCompactMaxStepTimeUs then the space will be reduced by factor of 2 (slow media),
    45 bet will never be less than KCompactMinStepSize.
    46 If the time is less than the KCompactMaxStepTimeUs/2 then the space will be increased by factor of 2 (fast media).
    47 
    48 @param aRemaining The remaining free database space.
    49 @param aStep The size of the space removed by the previous compaction step.
    50 @param aTime The execution time of the previous compaction step.
    51 
    52 @return The size in bytes if the next comaction step - the amount of space to be removed.
    53 
    54 @see KMaxStepTimeUs
    55 @see KMinStepSize
    56 @see KStartStepSize
    57 
    58 @internalComponent
    59 */
    60 static TInt CalcCompactionStep(TInt aRemaining, TInt aStep, TInt aTime)
    61 	{
    62 	if(aTime > KCompactMaxStepTimeUs)
    63 		{
    64 		aStep /= 2;
    65 		if(aStep < KCompactMinStepSize)
    66 			{
    67 			aStep = KCompactMinStepSize; 
    68 			}
    69 		}
    70 	else if(aTime <= (KCompactMaxStepTimeUs / 2))
    71 		{
    72 		aStep *= 2;
    73 		}
    74 	if(aRemaining < aStep)
    75 		{//If, for example, aStep is 4000 bytes, aRemaining is 2000 bytes, then the step should be 2000,  
    76 		 //because that is what is left in the database as a free space.
    77 		aStep = aRemaining;
    78 		}
    79 	return aStep;
    80 	}
    81  
    82 /**
    83 Creates a new CSqlDatabaseImpl instance.
    84 
    85 CSqlDatabaseImpl implements RSqlDatabase, which means that CSqlDatabaseImpl instance will be created from 
    86 RSqlDatabase functions doing RSqlDatabase instance initialization - Create() and Open().
    87 
    88 @param aFunction It may have one of the following values: 																																																																																																																																																																																																																																																																																																																																																																																																							
    89  ESqlSrvDbCreate       - Create a shared non-secure or private secure database;
    90  ESqlSrvDbCreateSecure - Create a shared secure database;
    91  ESqlSrvDbOpen         - Open a shared non-secure, shared secure or private secure database;
    92 	
    93 @param aDbFileName The name of the file that is to host the database.
    94 				   If it is a secure database, then the format of the name is 
    95 				   \<drive\>:\<[SID]database file name excluding the path\>. "[SID]" refers to the application SID.
    96 				   If it is a non-secure database then aDbFileName should contain the full path name of the file 
    97 				   that is to host the database.
    98 @param aSecurityPolicy The container for the security policies. 
    99 				   aSecurityPolicy is NULL if aDbFileName refers to a non-secure database.
   100 @param aConfig the configuration string "PARAM=VALUE;...."
   101 
   102 @return A pointer to the created CSqlDatabaseImpl instance.
   103 
   104 @leave KErrNoMemory, an out of memory condition has occurred;
   105        KErrBadName, the file name is invalid - it has either a zero length or it is the name of a directory;
   106        KErrArgument, system table name found in the security policies (aSecurityPolicy);
   107        KErrAlreadyExists, the file already exists;
   108        KErrNotReady, the drive does not exist or is not ready;
   109        KErrInUse, the file is already open;
   110        KErrNotFound, database file not found;
   111        KErrGeneral, missing or invalid security policies (if the database to be opened is a secure database);
   112        KErrNotSupported, incompatible SQL security version (if the database to be opened is a secure database).
   113        KErrPermissionDenied, the caller does not satisfy the relevant database security policies.
   114                       Note that the function may leave with database specific errors categorised as ESqlDbError and
   115                       other system-wide error codes.
   116                       
   117 @see RSqlDatabase
   118 @see RSqlDatabase::Create()
   119 @see RSqlDatabase::Open()
   120 @see TSqlSrvFunction
   121 @see CSqlSecurityPolicy
   122 */
   123 CSqlDatabaseImpl* CSqlDatabaseImpl::NewL(TSqlSrvFunction aFunction, const TDesC& aDbFileName, 
   124 										 const CSqlSecurityPolicy* aSecurityPolicy,
   125 										 const TDesC8* aConfig)
   126 	{
   127 	CSqlDatabaseImpl* self = new (ELeave) CSqlDatabaseImpl;
   128 	CleanupStack::PushL(self);
   129 	self->ConstructL(aFunction, aDbFileName, aSecurityPolicy, aConfig);
   130 	CleanupStack::Pop(self);
   131 	return self;
   132 	}
   133 
   134 /**
   135 Initializes the created CSqlDatabaseImpl instance.
   136 
   137 @param aFunction It may have one of the following values:
   138   ESqlSrvDbCreate       - Create a shared non-secure or private secure database;
   139   ESqlSrvDbCreateSecure - Create a shared secure database;
   140   ESqlSrvDbOpen         - Open a shared non-secure, shared secure or private secure database;
   141 
   142 @param aDbFileName The name of the file that is to host the database.
   143 				   If it is a secure database, then the format of the name is 
   144 				   \<drive\>:\<[SID]database file name excluding the path\>. "[SID]" refers to the application SID.
   145 				   If it is a non-secure database then aDbFileName should contain the full path name of the file 
   146 				   that is to host the database.
   147 @param aSecurityPolicy The container for the security policies. 
   148 				   aSecurityPolicy is NULL if aDbFileName refers to a non-secure database.
   149 @param aConfig the configuration string "PARAM=VALUE;...."
   150 
   151 @leave KErrNoMemory, an out of memory condition has occurred;
   152        KErrBadName, the file name is invalid - it has either a zero length or it is the name of a directory;
   153        KErrArgument, system table name found in the security policies (aSecurityPolicy);
   154        KErrAlreadyExists, the file already exists;
   155        KErrNotReady, the drive does not exist or is not ready;
   156        KErrInUse, the file is already open;
   157        KErrNotFound, database file not found;
   158        KErrGeneral, missing or invalid security policies (if the database to be opened is a secure database);
   159        KErrNotSupported, incompatible SQL security version (if the database to be opened is a secure database).
   160        KErrPermissionDenied, the caller does not satisfy the relevant database security policies.
   161                       Note that the function may leave with database specific errors categorised as ESqlDbError and
   162                       other system-wide error codes.
   163 
   164 @see CSqlDatabaseImpl::NewL()
   165 
   166 @panic SqlDb 4 In _DEBUG mode. aSecurityPolicy is NULL, but the request is for opening/creating a secure database.
   167 */
   168 void CSqlDatabaseImpl::ConstructL(TSqlSrvFunction aFunction, const TDesC& aDbFileName, 
   169 								  const CSqlSecurityPolicy* aSecurityPolicy, const TDesC8* aConfig)
   170 	{
   171 	TPtrC8 securityPolicyData;
   172 	if(aFunction == ESqlSrvDbCreateSecure)	
   173 		{
   174 		__ASSERT_DEBUG(aSecurityPolicy != NULL, __SQLPANIC(ESqlPanicBadArgument));
   175 		const RSqlBufFlat& bufFlat = aSecurityPolicy->BufFlat();
   176 		securityPolicyData.Set(bufFlat.BufDes());
   177 		}
   178 	__SQLLEAVE_IF_ERROR(Session().Connect(aFunction, aDbFileName, securityPolicyData, aConfig));
   179 	}
   180 
   181 /**
   182 Frees the allocated by CSqlDatabaseImpl instance memory and other resources.
   183 */
   184 CSqlDatabaseImpl::~CSqlDatabaseImpl()
   185 	{
   186 	Session().Close();
   187 	}
   188 
   189 /**
   190 Creates and returns a copy of the database security policies object.
   191 The caller is responsible for destroying the returned CSqlSecurityPolicy instance.
   192 
   193 Implements RSqlDatabase::GetSecurityPolicyL().
   194 
   195 @return A copy of the database security policies object. 
   196         The returned copy must be destroyed by the caller.
   197         
   198 @leave KErrNotSupported, the current database is not a secure database;
   199        KErrNoMemory, an out of memory condition has occurred;
   200 
   201 @see RSqlDatabase
   202 @see RSqlDatabase::GetSecurityPolicyL()
   203 */
   204 CSqlSecurityPolicy* CSqlDatabaseImpl::CloneSecurityPolicyL()
   205 	{
   206 	TSecurityPolicy defaultPolicy(TSecurityPolicy::EAlwaysFail);
   207 	CSqlSecurityPolicy* dbPolicy = CSqlSecurityPolicy::NewLC(defaultPolicy);
   208 	__SQLLEAVE_IF_ERROR(Session().GetSecurityPolicy(dbPolicy->BufFlat()));
   209 	CleanupStack::Pop(dbPolicy);
   210 	return dbPolicy;
   211 	}
   212 
   213 /**
   214 Implements RSqlDatabase::Compact().
   215 
   216 @param aSize Can be one of:
   217 				 RSqlDatabase::EMaxCompaction - requests a full database compaction. All free pages
   218 				  (if any exists) will be removed;
   219 				 Positive integer value - the server will attempt to compact the database removing 
   220 				  at most aSize bytes from the database file, rounded up to the nearest page count, 
   221 				  e.g. request for removing 1 byte will remove one free page from the database;
   222 @param aDbName The attached database name or KNullDesC for the main database
   223 
   224 @return Zero or positive integer - the operation has completed succesfully, the return value is the
   225          						   size of the removed free space in bytes,
   226 		KErrArgument, Invalid aSize value;
   227         KErrBadName, Invalid (too long) attached database name;
   228         KSqlErrReadOnly, Read-only database;
   229         KSqlErrGeneral, There is no an attached database with aDbName name;
   230                   Note that database specific errors categorised as ESqlDbError, and
   231                   other system-wide error codes may also be returned.
   232 
   233 Usage of the IPC call arguments:
   234 Arg 0: [out]	How much space in bytes should be compacted, all free pages should be removed if the
   235 				parameter value is RSqlDatabase::EMaxCompaction.
   236 Arg 1: [out]	The database name length in characters
   237 Arg 2: [out]	The attached database name or KNullDesC for the main database
   238 */
   239 TInt CSqlDatabaseImpl::Compact(TInt aSize, const TDesC& aDbName)
   240 	{
   241 	if(aSize < 0)
   242 		{
   243 		if(aSize != RSqlDatabase::EMaxCompaction)
   244 			{
   245 			return KErrArgument;
   246 			}
   247 		aSize = KMaxTInt;
   248 		}
   249 	TInt remaining = aSize;
   250 	TInt compacted = 0;
   251 	TInt step = KCompactStartStepSize;
   252 	TInt rc = 0;
   253 	TTimeIntervalMicroSeconds interval(0);
   254 	while(remaining > 0)
   255 		{
   256 		step = ::CalcCompactionStep(remaining, step, interval.Int64());
   257 		TTime start;
   258 		start.HomeTime();
   259 		rc = Session().SendReceive(ESqlSrvDbCompact, TIpcArgs(step, aDbName.Length(), &aDbName));
   260 		if(rc <= 0)
   261 			{
   262 			break;	
   263 			}
   264 		TTime end;
   265 		end.HomeTime();
   266 		interval = end.MicroSecondsFrom(start);
   267 		remaining -= rc;				
   268 		compacted += rc;
   269 		}
   270 	return rc < 0 ? rc : compacted;
   271 	}
   272 
   273 /**
   274 Usage of the IPC call arguments:
   275 Arg 0: [out]	How much space in bytes should be compacted, all free pages should be removed if the
   276 				parameter value is RSqlDatabase::EMaxCompaction.
   277 Arg 1: [out]	The database name length in characters
   278 Arg 2: [out]	The attached database name or KNullDesC for the main database
   279 */
   280 void CSqlDatabaseImpl::Compact(TInt aSize, const TDesC& aDbName, TRequestStatus& aStatus)
   281 	{
   282 	if(aSize == 0)
   283 		{
   284 		TRequestStatus* stat = &aStatus;
   285 		User::RequestComplete(stat, 0);
   286 		return;
   287 		}
   288 	Session().SendReceive(ESqlSrvDbCompact, TIpcArgs(aSize, aDbName.Length(), &aDbName), aStatus);
   289 	}