os/persistentdata/persistentstorage/dbms/sdbms/SD_SESS.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) 1998-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
// DBMS server-session and support classes
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include <s32file.h>
sl@0
    19
#include "SD_STD.H"
sl@0
    20
#include "Sd_DbList.h"
sl@0
    21
sl@0
    22
using namespace DBSC;
sl@0
    23
sl@0
    24
/*
sl@0
    25
CDbsSession class is updated now to with a support for DBMS security policies.
sl@0
    26
What idea sits behind the update:
sl@0
    27
1) DBMS server has a set of server side CDbsSession objects.
sl@0
    28
2) Each CDbsSession object has a list of CDbsSession::TEntry items.
sl@0
    29
   Each CDbsSession::TEntry item holds a server side DBMS object, which is
sl@0
    30
   used to perform client requests.
sl@0
    31
3) CDbsSession::TEntry object could be:
sl@0
    32
	- EDbsDatabase - database object. It operates at database level.
sl@0
    33
		OpenDatabaseL is the method, which creates it.
sl@0
    34
	- EDbsIncremental - incremental object. It operates either at database or table level.
sl@0
    35
		NewIncrementalL is the method, which creates it.
sl@0
    36
	- EDbsCursor - cursor object. It operates at table level.
sl@0
    37
		NewCursorL is the method, which creates it.
sl@0
    38
	- EDbsConstraint - constraint object. It is used at table level and needs the same
sl@0
    39
		security policy as its table.
sl@0
    40
	- EDbsStream - stream object. It operates either at database or table level.
sl@0
    41
		NewStreamL are the methods, which create it.
sl@0
    42
	- EDbsObserver - observer object. It operates at database level.
sl@0
    43
4) When DBMS server is used in a security environment, database/table security policies 
sl@0
    44
	describe allowed W/R operations for the database/tables.
sl@0
    45
5) Each database/table has a security policy object assiciated with it. It may be accessed
sl@0
    46
	using MPolicy interface.
sl@0
    47
6) There is a MPolicySpace interface, which holds a collection of MPolicy interfaces
sl@0
    48
	for the database/tables. MPolicy interfaces can be queried using MPolicySpace methods.
sl@0
    49
7) All DBMS server side objects, described in (3), operate at database/table level. In a
sl@0
    50
	security environment, the client's access rights has to be checked, before performing
sl@0
    51
	particular database/table operation.
sl@0
    52
	It is known at the moment of DBMS object creation, on which database/table it will operate.
sl@0
    53
	The database/table cannot be changed during the DBMS object life time.
sl@0
    54
	At the moment of DBMS object creation, the related database/table interface is queried and
sl@0
    55
	stored in CDbsSession::TEntry::iPolicy data member. 
sl@0
    56
	It is an error if CDbsSession::TEntry item exists without a security policy.
sl@0
    57
8) Every time, when a client needs particular DBMS server functionality and CDbsSession::ServiceL
sl@0
    58
	has been called, the session object will find which DBMS server object has to be used and
sl@0
    59
	asserts client's access rights against its security policy.
sl@0
    60
	If the client has not enough access rights to perform desired DBMS operation, the DBMS
sl@0
    61
	session will leave with KErrPermissionDenied error code.
sl@0
    62
	======================================================================================
sl@0
    63
	In one sentence - every time, when an object from CDbsSession::TEntry list is about to 
sl@0
    64
	perform particular operation at database/table level, the associated database/table 
sl@0
    65
	security policy will be checked and the opration may be rejected.
sl@0
    66
*/
sl@0
    67
sl@0
    68
// Class CDbsSession::TEntry
sl@0
    69
sl@0
    70
//
sl@0
    71
// release the object
sl@0
    72
//
sl@0
    73
void CDbsSession::TEntry::Release()
sl@0
    74
	{
sl@0
    75
	TDbsType t=Type();
sl@0
    76
	__ASSERT(t!=EDbsFree);
sl@0
    77
	if (t==EDbsObserver)
sl@0
    78
		delete STATIC_CAST(CDbsObserver::HObserver*,iObject);
sl@0
    79
	else if (t==EDbsStream)
sl@0
    80
		delete STATIC_CAST(HDbsStream*,iObject);
sl@0
    81
	else
sl@0
    82
		CDbObject::Destroy(STATIC_CAST(CDbObject*,iObject));
sl@0
    83
	iType=TUint8(EDbsFree);
sl@0
    84
	iPolicy = NULL;
sl@0
    85
	}
sl@0
    86
sl@0
    87
#ifdef __DBDUMP__
sl@0
    88
//Using CDbsSession::TEntry::Dump() method you can dump the TEntry's content
sl@0
    89
//into a stream. Note that the dump works only if you have __DBDUMP__ macro defined.
sl@0
    90
void CDbsSession::TEntry::Dump(RFile& aFile) const
sl@0
    91
	{
sl@0
    92
	_LIT8(KClassName, "CDbsSession::TEntry. this=%X");
sl@0
    93
	_LIT8(KType, "Type=%S");
sl@0
    94
	_LIT8(KMagic, "Magic=%d");
sl@0
    95
	_LIT8(KCrLf, "\r\n");
sl@0
    96
	_LIT8(KEnd, "====");
sl@0
    97
	TPtrC8 KDbsType[EDbsMaxType + 1] = {_L8("Free"), _L8("Database"), _L8("Incremental"), _L8("Cursor"), _L8("Constraint"), _L8("Stream"), _L8("Observer")};
sl@0
    98
	TBuf8<40> buf;
sl@0
    99
sl@0
   100
	buf.Format(KClassName, this);
sl@0
   101
	(void)aFile.Write(buf);
sl@0
   102
	(void)aFile.Write(KCrLf);
sl@0
   103
	buf.Format(KType, &KDbsType[iType]);
sl@0
   104
	(void)aFile.Write(buf);
sl@0
   105
	(void)aFile.Write(KCrLf);
sl@0
   106
	buf.Format(KMagic, iMagic);
sl@0
   107
	(void)aFile.Write(buf);
sl@0
   108
	(void)aFile.Write(KCrLf);
sl@0
   109
	__ASSERT(iPolicy);
sl@0
   110
	iPolicy->Dump(aFile);
sl@0
   111
	(void)aFile.Write(KEnd);
sl@0
   112
	(void)aFile.Write(KCrLf);
sl@0
   113
	}
sl@0
   114
#endif//__DBDUMP__
sl@0
   115
sl@0
   116
// Class CDbsSession
sl@0
   117
sl@0
   118
//
sl@0
   119
// Release all this clients resources
sl@0
   120
// Streams must be released before cursors, and everything before the connection
sl@0
   121
//
sl@0
   122
CDbsSession::~CDbsSession()
sl@0
   123
	{
sl@0
   124
	TEntry* const base=&iIx[0];
sl@0
   125
	if (base)
sl@0
   126
		{
sl@0
   127
		TInt type=EDbsMaxType;
sl@0
   128
		do	{
sl@0
   129
			for (TEntry* e=base+iSize;--e>=base;)
sl@0
   130
				{
sl@0
   131
				if (e->iType==type)
sl@0
   132
					e->Release();
sl@0
   133
				}
sl@0
   134
			} while (--type>EDbsFree);
sl@0
   135
		User::Free(base);
sl@0
   136
		}
sl@0
   137
	iDbPolicyRqColl.Close();
sl@0
   138
    delete iSessDriveSpace;
sl@0
   139
	Server().RemoveSession();
sl@0
   140
	}
sl@0
   141
sl@0
   142
/**
sl@0
   143
Overrides virtual CSession2::Create2().
sl@0
   144
Creates iSessDriveSpace instance.
sl@0
   145
*/
sl@0
   146
void CDbsSession::CreateL()
sl@0
   147
    {
sl@0
   148
    iSessDriveSpace = CDbsSessDriveSpace::NewL(Server().DriveSpaceCol());
sl@0
   149
    }
sl@0
   150
sl@0
   151
//
sl@0
   152
// provide for CSession
sl@0
   153
// If this leaves, we complete message through ServiceError(...)
sl@0
   154
//
sl@0
   155
void CDbsSession::ServiceL(const RMessage2& aMessage)
sl@0
   156
	{
sl@0
   157
	//set received message in server so it can be used in cleanup when server panics.
sl@0
   158
	CPolicyProxy& policyProxy = Server().PolicyProxy();
sl@0
   159
	TDbsFunction f=DbsFunction(aMessage.Function());
sl@0
   160
	if (f&KDbsObjectReturn)		// may return a derived object
sl@0
   161
		AllocL();	// ensure index space
sl@0
   162
	TInt r=0;//object handle
sl@0
   163
	TInt h=DbsHandle(aMessage.Function());
sl@0
   164
	if (h==KDbsSessionHandle)
sl@0
   165
		{	// session based requests
sl@0
   166
		switch (f&~KDbsObjectReturn)
sl@0
   167
			{
sl@0
   168
		case EDbsResourceMark:
sl@0
   169
			ResourceCountMarkStart();
sl@0
   170
			break;
sl@0
   171
		case EDbsResourceCheck:
sl@0
   172
			ResourceCountMarkEnd(aMessage);
sl@0
   173
			break;
sl@0
   174
		case EDbsResourceCount:
sl@0
   175
			r=CountResources();
sl@0
   176
			break;
sl@0
   177
		case EDbsSetHeapFailure:
sl@0
   178
			User::__DbgSetAllocFail(RHeap::EUser,RHeap::TAllocFail(aMessage.Int0()),aMessage.Int1());
sl@0
   179
			break;
sl@0
   180
		case EDbsOpenDatabase:
sl@0
   181
			r=OpenDatabaseL(aMessage);
sl@0
   182
			break;
sl@0
   183
		case EDbsReserveDriveSpace:
sl@0
   184
			ReserveDriveSpaceL(static_cast <TDriveNumber> (aMessage.Int0()));
sl@0
   185
			break;
sl@0
   186
        case EDbsFreeReservedSpace:
sl@0
   187
	        FreeReservedSpace(static_cast <TDriveNumber> (aMessage.Int0()));
sl@0
   188
			break;
sl@0
   189
		case EDbsReserveGetAccess:
sl@0
   190
			GetReserveAccessL(static_cast <TDriveNumber> (aMessage.Int0()));
sl@0
   191
			break;
sl@0
   192
		case EDbsReserveReleaseAccess:
sl@0
   193
			ReleaseReserveAccess(static_cast <TDriveNumber> (aMessage.Int0()));
sl@0
   194
			break;
sl@0
   195
		default:
sl@0
   196
			r=ExtServiceL(aMessage, static_cast <TDbsFunction> (f&~KDbsObjectReturn));
sl@0
   197
			break;
sl@0
   198
			}
sl@0
   199
		}
sl@0
   200
	else
sl@0
   201
		{	// object based requests
sl@0
   202
		TEntry& e=Object(h);
sl@0
   203
		__ASSERT(e.iPolicy);
sl@0
   204
		policyProxy.CheckL(aMessage, *e.iPolicy);
sl@0
   205
		TDbsFunction dbmsFuncNo = static_cast <TDbsFunction> (f & ~KDbsObjectReturn);
sl@0
   206
		switch (dbmsFuncNo)
sl@0
   207
			{
sl@0
   208
	// the common close function
sl@0
   209
		case EDbsClose:
sl@0
   210
			//If this is a database object, remove the related pair (handle, uid) from iDbPolicyRqColl collection.
sl@0
   211
			if(e.Type() == EDbsDatabase)
sl@0
   212
				{
sl@0
   213
				iDbPolicyRqColl.Remove(h);
sl@0
   214
				}
sl@0
   215
			Free(e);
sl@0
   216
			break;
sl@0
   217
	// database functions
sl@0
   218
		case EDbsDatabaseAuthenticate:
sl@0
   219
			{
sl@0
   220
            __LEAVE(KErrNotSupported);             
sl@0
   221
			}
sl@0
   222
			break;
sl@0
   223
		case EDbsDatabaseDestroy:
sl@0
   224
			r=e.Database().Destroy();
sl@0
   225
			break;
sl@0
   226
		case EDbsDatabaseBegin:
sl@0
   227
			r=e.Database().Begin();
sl@0
   228
			break;
sl@0
   229
		case EDbsDatabaseCommit:
sl@0
   230
			r=e.Database().Commit();
sl@0
   231
			break;
sl@0
   232
		case EDbsDatabaseRollback:
sl@0
   233
			e.Database().Rollback();
sl@0
   234
			break;
sl@0
   235
		case EDbsDatabaseProperty:
sl@0
   236
			r=e.Database().Property(CDbDatabase::TProperty(aMessage.Int0()));
sl@0
   237
			break;
sl@0
   238
		case EDbsDatabaseTables:
sl@0
   239
			{//The new stream object will have the same security policy as the database
sl@0
   240
			 //Read only stream
sl@0
   241
			CDbTableNames* names=CDbTableNames::NewLC();
sl@0
   242
			e.Database().TablesL(*names);
sl@0
   243
			r=NewStreamL(names,Externalizer(names),aMessage,e.iPolicy);
sl@0
   244
			}
sl@0
   245
			break;
sl@0
   246
		case EDbsDatabaseColumns:
sl@0
   247
			{//The new stream object will have the same security policy as the database
sl@0
   248
			 //Read only stream
sl@0
   249
			CDbColSet* set=CDbColSet::NewLC();
sl@0
   250
			e.Database().ColumnsL(*set,ReadName0L(0,aMessage));
sl@0
   251
			r=NewStreamL(set,Externalizer(set),aMessage,e.iPolicy);
sl@0
   252
			}
sl@0
   253
			break;
sl@0
   254
		case EDbsDatabaseIndexes:
sl@0
   255
			{//The new stream object will have the same security policy as the database
sl@0
   256
			 //Read only stream
sl@0
   257
			CDbIndexNames* names=CDbIndexNames::NewLC();
sl@0
   258
			e.Database().IndexesL(*names,ReadName0L(0,aMessage));
sl@0
   259
			r=NewStreamL(names,Externalizer(names),aMessage,e.iPolicy);
sl@0
   260
			}
sl@0
   261
			break;
sl@0
   262
		case EDbsDatabaseKeys:
sl@0
   263
			{//The new stream object will have the same security policy as the database
sl@0
   264
			 //Read only stream
sl@0
   265
			CDbKey* key=CDbKey::NewLC();
sl@0
   266
			e.Database().KeysL(*key,ReadName0L(0,aMessage),ReadName1L(1,aMessage));
sl@0
   267
			r=NewStreamL(key,Externalizer(key),aMessage,e.iPolicy);
sl@0
   268
			}
sl@0
   269
			break;
sl@0
   270
		case EDbsDatabaseCreateTable:
sl@0
   271
			{
sl@0
   272
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   273
			CDbColSet* set=ColSetLC(1,aMessage);
sl@0
   274
			CDbKey* primary=0;
sl@0
   275
			if (aMessage.Ptr2())
sl@0
   276
				primary=KeyLC(2,aMessage);
sl@0
   277
sl@0
   278
			e.Database().CreateTableL(name,*set,primary);
sl@0
   279
			if (primary)
sl@0
   280
				CleanupStack::PopAndDestroy();	// primary
sl@0
   281
			CleanupStack::PopAndDestroy();	// set
sl@0
   282
			}
sl@0
   283
			break;
sl@0
   284
		case EDbsDatabaseOpenObserver:
sl@0
   285
			//The new observer object will have the same security policy as the database
sl@0
   286
			r=Add(CDbsConnection::Source(e.Database()).ObserverL(),e.iPolicy);
sl@0
   287
			break;
sl@0
   288
		case EDbsDatabaseOpenUtility:
sl@0
   289
			{//The new incremental object will have the same security policy as the database
sl@0
   290
			TInt step;
sl@0
   291
			r=NewIncrementalL(e.Database().UtilityL(CDbDatabase::TUtility(aMessage.Int0()),step),step,aMessage,e.iPolicy);
sl@0
   292
			}
sl@0
   293
			break;
sl@0
   294
		case EDbsDatabaseOpenDropTable:
sl@0
   295
			{//The new incremental object will have the same security policy as the database
sl@0
   296
			TInt step;
sl@0
   297
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   298
			r=NewIncrementalL(e.Database().DropTableL(name,step),step,aMessage,e.iPolicy);
sl@0
   299
			}
sl@0
   300
			break;
sl@0
   301
		case EDbsDatabaseOpenAlterTable:
sl@0
   302
			{//The new incremental object will have the same security policy as the database
sl@0
   303
			TInt step;
sl@0
   304
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   305
			CDbColSet* set=ColSetLC(1,aMessage);
sl@0
   306
			r=NewIncrementalL(e.Database().AlterTableL(name,*set,step),step,aMessage,e.iPolicy);
sl@0
   307
			CleanupStack::PopAndDestroy();	// set
sl@0
   308
			}
sl@0
   309
			break;
sl@0
   310
		case EDbsDatabaseOpenCreateIndex:
sl@0
   311
			{//The new incremental object will have the same security policy as the database
sl@0
   312
			TInt step;
sl@0
   313
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   314
			const TDesC& table=ReadName1L(1,aMessage);
sl@0
   315
			CDbKey* key=KeyLC(2,aMessage);
sl@0
   316
			r=NewIncrementalL(e.Database().CreateIndexL(name,table,*key,step),step,aMessage,e.iPolicy);
sl@0
   317
			CleanupStack::PopAndDestroy();	// key
sl@0
   318
			}
sl@0
   319
			break;
sl@0
   320
		case EDbsDatabaseOpenDropIndex:
sl@0
   321
			{//The new incremental object will have the same security policy as the database
sl@0
   322
			TInt step;
sl@0
   323
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   324
			const TDesC& table=ReadName1L(1,aMessage);
sl@0
   325
			r=NewIncrementalL(e.Database().DropIndexL(name,table,step),step,aMessage,e.iPolicy);
sl@0
   326
			}
sl@0
   327
			break;
sl@0
   328
		case EDbsDatabaseExecute:
sl@0
   329
			{//The new incremental object will have the same security policy as the database or table - sql dependent.
sl@0
   330
			//In order EDbsDatabaseExecute request to be performed, the DBMS session has already
sl@0
   331
			//checked the client capabilities at the database level.
sl@0
   332
			//Now, the sql string will be parsed and the client capabilities will be checked
sl@0
   333
			//again at more specific database or table level.
sl@0
   334
			TInt init;
sl@0
   335
			HBufC* sql=ReadHBufLC(0,aMessage);
sl@0
   336
			const TDbPolicyRequest& rq = iDbPolicyRqColl[h];
sl@0
   337
			TPolicyType policyType = EPTNone;
sl@0
   338
			const MPolicy* policy = policyProxy.SqlPolicyL(rq, *sql, policyType);
sl@0
   339
			policyProxy.CheckL(policyType, aMessage, *policy);			
sl@0
   340
			r=NewIncrementalL(e.Database().ExecuteL(*sql,TDbTextComparison(aMessage.Int1()),init),init,aMessage,policy);
sl@0
   341
			CleanupStack::PopAndDestroy(sql);
sl@0
   342
			}
sl@0
   343
			break;
sl@0
   344
		case EDbsDatabasePrepareView:
sl@0
   345
			{//The new cursor object will have the same security policy as the table in the sql string.
sl@0
   346
			//In order EDbsDatabasePrepareView request to be performed, the DBMS session has already
sl@0
   347
			//checked the client capabilities at the database level.
sl@0
   348
			//Now, the sql string will be parsed and the client capabilities will be checked
sl@0
   349
			//again at table level.
sl@0
   350
			HBufC* sql=ReadHBufLC(0,aMessage);
sl@0
   351
			const TDbPolicyRequest& rq = iDbPolicyRqColl[h];
sl@0
   352
			TPolicyType policyType = EPTNone;
sl@0
   353
			const MPolicy* policy = policyProxy.SqlPolicyL(rq, *sql, policyType);
sl@0
   354
			policyProxy.CheckL(policyType, aMessage, *policy);			
sl@0
   355
			TPckgBuf<TDbWindow> window;
sl@0
   356
			aMessage.ReadL(2,window);
sl@0
   357
			TInt mode=aMessage.Int1();
sl@0
   358
			CDbCursor* cursor=e.Database().ViewL(TDbQuery(*sql,TDbTextComparison(mode&0xff)),window(),RDbRowSet::TAccess(mode>>8));
sl@0
   359
			r=NewCursorL(cursor,aMessage,policy);
sl@0
   360
			CleanupStack::PopAndDestroy(sql);
sl@0
   361
			}
sl@0
   362
			break;
sl@0
   363
		case EDbsDatabaseOpenTable:
sl@0
   364
			{//The new cursor object will have the same security policy as the table
sl@0
   365
			const TDesC& name=ReadName0L(0,aMessage);
sl@0
   366
			const TDbPolicyRequest& rq = iDbPolicyRqColl[h];
sl@0
   367
			const MPolicy* policy = policyProxy.TblPolicyL(rq, name);
sl@0
   368
			//Check caller capabilities against the table policy
sl@0
   369
			RDbRowSet::TAccess accessType = static_cast <RDbRowSet::TAccess> (aMessage.Int1());
sl@0
   370
			TPolicyType policyType = accessType == RDbRowSet::EReadOnly ? EPTRead : EPTWrite;
sl@0
   371
			policyProxy.CheckL(policyType, aMessage, *policy);
sl@0
   372
			//Create cursor
sl@0
   373
			r=NewCursorL(e.Database().TableL(name,accessType),aMessage,policy);
sl@0
   374
			}
sl@0
   375
			break;
sl@0
   376
	// Incremental functions
sl@0
   377
		case EDbsIncrementalNext:
sl@0
   378
			{
sl@0
   379
			TPckgBuf<TInt> step=aMessage.Int0();
sl@0
   380
			r=e.Incremental().NextL(step());
sl@0
   381
			aMessage.WriteL(1,step);
sl@0
   382
			}
sl@0
   383
			break;
sl@0
   384
	// Observer functions
sl@0
   385
		case EDbsObserverNotify:
sl@0
   386
			e.Observer().Notify(aMessage);
sl@0
   387
			return;		// deferred completion of the message!
sl@0
   388
		case EDbsObserverCancel:
sl@0
   389
			e.Observer().Cancel();
sl@0
   390
			break;
sl@0
   391
	// cursor functions
sl@0
   392
		case EDbsCursorColumnTypes:
sl@0
   393
			{
sl@0
   394
			TInt count=e.Cursor().ColumnCount();
sl@0
   395
			TUint8* types=(TUint8*)User::AllocLC(count);
sl@0
   396
			for (TInt ii=count;--ii>=0;)
sl@0
   397
				types[ii]=TUint8(e.Cursor().ColumnType(ii+1));
sl@0
   398
			aMessage.WriteL(3,TPtrC8(types,count));
sl@0
   399
			CleanupStack::PopAndDestroy();
sl@0
   400
			}
sl@0
   401
			break;
sl@0
   402
		case EDbsCursorReset:
sl@0
   403
			e.Cursor().Reset();
sl@0
   404
			break;
sl@0
   405
		case EDbsCursorEvaluate:
sl@0
   406
			r=e.Cursor().EvaluateL()?1:0;
sl@0
   407
			break;
sl@0
   408
		case EDbsCursorUnevaluated:
sl@0
   409
			r=e.Cursor().Unevaluated()?1:0;
sl@0
   410
			break;
sl@0
   411
		case EDbsCursorSetIndex:
sl@0
   412
			{
sl@0
   413
			const TDesC* name=0;
sl@0
   414
			if (aMessage.Ptr0())
sl@0
   415
				name=&ReadName0L(0,aMessage);
sl@0
   416
			e.Cursor().SetIndexL(name);
sl@0
   417
			}
sl@0
   418
			break;
sl@0
   419
		case EDbsCursorSeek:
sl@0
   420
			{
sl@0
   421
			TDbLookupKey* key=LookupKeyLC(0,aMessage.Int1(),aMessage);
sl@0
   422
			r=e.Cursor().SeekL(*key,RDbTable::TComparison(aMessage.Int2()))?1:0;
sl@0
   423
			CleanupStack::PopAndDestroy();		// key;
sl@0
   424
			}
sl@0
   425
			break;
sl@0
   426
		case EDbsCursorAtBeginning:
sl@0
   427
			r=e.Cursor().AtBeginning()?1:0;
sl@0
   428
			break;
sl@0
   429
		case EDbsCursorAtEnd:
sl@0
   430
			r=e.Cursor().AtEnd()?1:0;
sl@0
   431
			break;
sl@0
   432
		case EDbsCursorAtRow:
sl@0
   433
			r=e.Cursor().AtRow()?1:0;
sl@0
   434
			break;
sl@0
   435
		case EDbsCursorCount:
sl@0
   436
			r=e.Cursor().CountL(RDbRowSet::TAccuracy(aMessage.Int0()))+1;
sl@0
   437
			break;
sl@0
   438
		case EDbsCursorGotoPos:
sl@0
   439
			if (e.Cursor().GotoL(RDbRowSet::TPosition(aMessage.Int0())))
sl@0
   440
				{
sl@0
   441
				e.Cursor().GetL();
sl@0
   442
				r=RetrieveRowL(e.Cursor(),aMessage);
sl@0
   443
				}
sl@0
   444
			break;
sl@0
   445
		case EDbsCursorBookmark:
sl@0
   446
			{
sl@0
   447
			TPckgBuf<TDbBookmark::TMark> mark;
sl@0
   448
			e.Cursor().Bookmark(mark());
sl@0
   449
			aMessage.WriteL(3,mark);
sl@0
   450
			}
sl@0
   451
			break;
sl@0
   452
		case EDbsCursorGotoBookmark:
sl@0
   453
			{
sl@0
   454
			TPckgBuf<TDbBookmark::TMark> mark;
sl@0
   455
			aMessage.ReadL(0,mark);
sl@0
   456
			e.Cursor().GotoL(mark());
sl@0
   457
			}
sl@0
   458
			break;
sl@0
   459
		case EDbsCursorGet:
sl@0
   460
			e.Cursor().GetL();
sl@0
   461
			r=RetrieveRowL(e.Cursor(),aMessage);
sl@0
   462
			break;
sl@0
   463
		case EDbsCursorInsert:
sl@0
   464
			e.Cursor().InsertL(CDbCursor::TInsert(aMessage.Int0()));
sl@0
   465
			r=RetrieveRowL(e.Cursor(),aMessage);
sl@0
   466
			break;
sl@0
   467
		case EDbsCursorUpdate:
sl@0
   468
			e.Cursor().UpdateL();
sl@0
   469
			r=RetrieveRowL(e.Cursor(),aMessage);
sl@0
   470
			break;
sl@0
   471
		case EDbsCursorRetrieveRow:	// pass the row buffer back to the client
sl@0
   472
			r=RetrieveRowL(e.Cursor(),aMessage);
sl@0
   473
			break;
sl@0
   474
		case EDbsCursorCancel:
sl@0
   475
			e.Cursor().Cancel();
sl@0
   476
			break;
sl@0
   477
		case EDbsCursorPut:
sl@0
   478
			PutRowL(e.Cursor(),aMessage);
sl@0
   479
			e.Cursor().PutL();
sl@0
   480
			break;
sl@0
   481
		case EDbsCursorDelete:
sl@0
   482
			e.Cursor().DeleteL();
sl@0
   483
			break;
sl@0
   484
		case EDbsCursorColumns:
sl@0
   485
			{// reduce memory usage by extracting and stream columns individually
sl@0
   486
			 //Read only stream
sl@0
   487
			RWriteStream strm(HBufBuf::NewLC());
sl@0
   488
			TInt count=e.Cursor().ColumnCount();
sl@0
   489
			strm.WriteInt32L(count);
sl@0
   490
			TDbCol col;
sl@0
   491
			for (TInt ii=0;++ii<=count;)
sl@0
   492
				{
sl@0
   493
				e.Cursor().ColumnDef(col,ii);
sl@0
   494
				strm<<col;
sl@0
   495
				}
sl@0
   496
			strm.CommitL();
sl@0
   497
			TInt ext=strm.Sink()->SizeL();
sl@0
   498
			CleanupStack::Pop();			// stream
sl@0
   499
			r=NewStreamL(strm.Sink(),aMessage,e.iPolicy,ext);
sl@0
   500
			}
sl@0
   501
			break;
sl@0
   502
		case EDbsCursorColumnDef:
sl@0
   503
			{
sl@0
   504
			TPckgBuf<TDbCol> col;
sl@0
   505
			e.Cursor().ColumnDef(col(),aMessage.Int0());
sl@0
   506
			aMessage.WriteL(3,col);
sl@0
   507
			}
sl@0
   508
			break;
sl@0
   509
		case EDbsCursorSetNull:
sl@0
   510
			__ASSERT(TDbCol::IsLong(e.Cursor().ColumnType(aMessage.Int0())));
sl@0
   511
			e.Cursor().SetNullL(aMessage.Int0());
sl@0
   512
			break;
sl@0
   513
		case EDbsCursorColumnSize:
sl@0
   514
			__ASSERT(TDbCol::IsLong(e.Cursor().ColumnType(aMessage.Int0())));
sl@0
   515
			r=e.Cursor().ColumnSize(aMessage.Int0());
sl@0
   516
			break;
sl@0
   517
		case EDbsCursorColumnSource:
sl@0
   518
			__ASSERT(TDbCol::IsLong(e.Cursor().ColumnType(aMessage.Int0())));
sl@0
   519
			{//The new stream object will have the same security policy as the table
sl@0
   520
			 //Read only stream, used for large BLOB fields.
sl@0
   521
			TDbColNo col=aMessage.Int0();
sl@0
   522
			CDbCursor& cursor=e.Cursor();
sl@0
   523
			r=NewStreamL(cursor.ColumnSourceL(col),aMessage,e.iPolicy,cursor.ColumnSize(col));
sl@0
   524
			}
sl@0
   525
			break;
sl@0
   526
		case EDbsCursorColumnSink:
sl@0
   527
			//The new stream object will have the same security policy as the table
sl@0
   528
			//Write stream, used for large BLOB fields.
sl@0
   529
			r=NewStreamL(e.Cursor().ColumnSinkL(aMessage.Int0()),aMessage,e.iPolicy);
sl@0
   530
			break;
sl@0
   531
		case EDbsCursorOpenConstraint:
sl@0
   532
			{//The new constraint object will have the same security policy as the table
sl@0
   533
			HBufC* sql=ReadHBufLC(0,aMessage);
sl@0
   534
			CDbRowConstraint* ct=e.Cursor().ConstraintL(TDbQuery(*sql,TDbTextComparison(aMessage.Int1())));
sl@0
   535
			CleanupStack::PopAndDestroy();	//sql
sl@0
   536
			r=Add(ct, e.iPolicy);
sl@0
   537
			}
sl@0
   538
			break;
sl@0
   539
		case EDbsCursorMatch:
sl@0
   540
			r=e.Cursor().MatchL(Object(aMessage.Int0()).Constraint())?1:0;
sl@0
   541
			break;
sl@0
   542
		case EDbsCursorFind:
sl@0
   543
			{
sl@0
   544
			HBufC* sql=ReadHBufLC(0,aMessage);
sl@0
   545
			TDbQuery q(*sql,TDbTextComparison(aMessage.Int1()));
sl@0
   546
			r=e.Cursor().FindL(RDbRowSet::TDirection(aMessage.Int2()),q)-KErrNotFound;
sl@0
   547
			CleanupStack::PopAndDestroy();	//sql
sl@0
   548
			}
sl@0
   549
			break;
sl@0
   550
			// stream functions
sl@0
   551
		case EDbsStreamRead:
sl@0
   552
			{
sl@0
   553
			r=e.Stream().ReadL(aMessage);
sl@0
   554
			}
sl@0
   555
			break;
sl@0
   556
		case EDbsStreamWrite:
sl@0
   557
			{
sl@0
   558
			e.Stream().WriteL(aMessage);
sl@0
   559
			}
sl@0
   560
			break;
sl@0
   561
		case EDbsStreamSize:
sl@0
   562
			{
sl@0
   563
			r=e.Stream().SizeL();
sl@0
   564
			}
sl@0
   565
			break;
sl@0
   566
		case EDbsStreamSynch:
sl@0
   567
			{
sl@0
   568
			e.Stream().SynchL();
sl@0
   569
			}
sl@0
   570
			break;
sl@0
   571
	// we don't know about any other functions
sl@0
   572
		default:
sl@0
   573
			r=KErrNotSupported;
sl@0
   574
			break;
sl@0
   575
			}
sl@0
   576
		}
sl@0
   577
sl@0
   578
	// don't complete message if we have paniced client, just reset variable;
sl@0
   579
	if (!aMessage.IsNull())
sl@0
   580
		aMessage.Complete(r);
sl@0
   581
	}
sl@0
   582
sl@0
   583
sl@0
   584
// Handle an error from CDbsSession::ServiceL()
sl@0
   585
// A bad descriptor error implies a badly programmed client, so panic it;
sl@0
   586
// otherwise use the default handling (report the error to the client)
sl@0
   587
void CDbsSession::ServiceError(const RMessage2& aMessage,TInt aError)
sl@0
   588
 	{
sl@0
   589
#ifdef __DBDUMP__
sl@0
   590
	Dump();
sl@0
   591
#endif//__DBDUMP__
sl@0
   592
 	if (aError==KErrBadDescriptor)
sl@0
   593
 		{
sl@0
   594
 		// this completes message as well so we have to handle it in
sl@0
   595
 		// ServiceL
sl@0
   596
 		_LIT(KCategory,"DBMS-server");
sl@0
   597
 		Server().Panic(KCategory,EDbsBadDescriptor);
sl@0
   598
 		}
sl@0
   599
 	CSession2::ServiceError(aMessage,aError);
sl@0
   600
   	}
sl@0
   601
sl@0
   602
// Read aPtr as a name
sl@0
   603
const TDesC& CDbsSession::ReadName0L(TInt aIndex,const RMessage2& aMessage)
sl@0
   604
	{
sl@0
   605
	TDes& name=Server().Name0();
sl@0
   606
	aMessage.ReadL(aIndex,name);
sl@0
   607
	return name;
sl@0
   608
	}
sl@0
   609
sl@0
   610
// Read Ptr1 as a name
sl@0
   611
const TDesC& CDbsSession::ReadName1L(TInt aIndex,const RMessage2& aMessage)
sl@0
   612
	{
sl@0
   613
	TDes& name=Server().Name1();
sl@0
   614
	aMessage.ReadL(aIndex,name);
sl@0
   615
	return name;
sl@0
   616
	}
sl@0
   617
sl@0
   618
// Reads aPtr as a variable length 8-bit descriptor
sl@0
   619
HBufC8* CDbsSession::ReadHBuf8LC(TInt aIndex,const RMessage2& aMessage)
sl@0
   620
	{
sl@0
   621
	TInt len= aMessage.GetDesLengthL(aIndex);
sl@0
   622
	HBufC8* buf=HBufC8::NewLC(len);
sl@0
   623
	TPtr8 read(buf->Des());
sl@0
   624
	aMessage.ReadL(aIndex,read);
sl@0
   625
	return buf;
sl@0
   626
	}
sl@0
   627
sl@0
   628
// Reads aPtr as a variable length build-width descriptor
sl@0
   629
HBufC* CDbsSession::ReadHBufLC(TInt aIndex,const RMessage2& aMessage)
sl@0
   630
	{
sl@0
   631
	TInt len= aMessage.GetDesLengthL(aIndex);
sl@0
   632
	HBufC* buf=HBufC::NewLC(len);
sl@0
   633
	TPtr read(buf->Des());
sl@0
   634
	aMessage.ReadL(aIndex,read);
sl@0
   635
	return buf;
sl@0
   636
	}
sl@0
   637
sl@0
   638
// Pulls a column set from the client
sl@0
   639
CDbColSet* CDbsSession::ColSetLC(TInt aIndex,const RMessage2& aMessage)
sl@0
   640
	{
sl@0
   641
	CDbColSet* set=CDbColSet::NewLC();
sl@0
   642
	HBufC8* data=ReadHBuf8LC(aIndex,aMessage);
sl@0
   643
	RDesReadStream strm(*data);
sl@0
   644
	strm>>*set;
sl@0
   645
	CleanupStack::PopAndDestroy();	// data
sl@0
   646
	return set;
sl@0
   647
	}
sl@0
   648
sl@0
   649
// Pulls a key from the client
sl@0
   650
CDbKey* CDbsSession::KeyLC(TInt aIndex,const RMessage2& aMessage)
sl@0
   651
	{
sl@0
   652
	CDbKey* key=CDbKey::NewLC();
sl@0
   653
	HBufC8* data=ReadHBuf8LC(aIndex,aMessage);
sl@0
   654
	RDesReadStream strm(*data);
sl@0
   655
	strm>>*key;
sl@0
   656
	CleanupStack::PopAndDestroy();	// data
sl@0
   657
	return key;
sl@0
   658
	}
sl@0
   659
sl@0
   660
// Reconstructs a TDbLookupKey from the message parameter
sl@0
   661
TDbLookupKey* CDbsSession::LookupKeyLC(TInt aIndex,TInt aSize,const RMessage2& aMessage)
sl@0
   662
	{
sl@0
   663
	TDbLookupKey* key=(TDbLookupKey*)User::AllocLC(aSize);
sl@0
   664
	TPtr8 pckg((TUint8*)key,aSize);
sl@0
   665
	aMessage.ReadL(aIndex,pckg); 
sl@0
   666
	TDbLookupKey::SColumn* iter=CONST_CAST(TDbLookupKey::SColumn*,key->First());
sl@0
   667
	const TDbLookupKey::SColumn* end=iter+key->Count();
sl@0
   668
	if ((TUint8*)end-(TUint8*)key<aSize)
sl@0
   669
		{	// there is some text data following...
sl@0
   670
		do
sl@0
   671
			{
sl@0
   672
			switch (iter->iType)
sl@0
   673
				{
sl@0
   674
			default:
sl@0
   675
				break;
sl@0
   676
			case EDbColText8:
sl@0
   677
				iter->iDes8.iPtr=(TUint8*)key+TInt(iter->iDes8.iPtr);
sl@0
   678
				break;
sl@0
   679
			case EDbColText16:
sl@0
   680
				iter->iDes16.iPtr=(TUint16*)((TUint8*)key+TInt(iter->iDes8.iPtr));
sl@0
   681
				break;
sl@0
   682
				};
sl@0
   683
			} while (++iter<end);
sl@0
   684
		}
sl@0
   685
	return key;
sl@0
   686
	}
sl@0
   687
sl@0
   688
// Reads the row from the client buffer into the server-side buffer
sl@0
   689
void CDbsSession::ReadRowL(RDbRow& aRow,TInt aSize,const RMessage2& aMessage)
sl@0
   690
	{
sl@0
   691
	if (aSize)
sl@0
   692
		{
sl@0
   693
		aRow.GrowL(aSize);
sl@0
   694
		TPtr8 buf((TUint8*)aRow.First(),aSize);
sl@0
   695
		aMessage.ReadL(0,buf);
sl@0
   696
		}
sl@0
   697
	aRow.SetSize(aSize);
sl@0
   698
	}
sl@0
   699
sl@0
   700
// Writes a row from the client into the cursor
sl@0
   701
// Any blobs which are modified must not be transferred into the cursor row buffer
sl@0
   702
void CDbsSession::PutRowL(CDbCursor& aCursor,const RMessage2& aMessage)
sl@0
   703
	{
sl@0
   704
	RDbRow* pRow=aCursor.RowBuffer();
sl@0
   705
	TInt size= aMessage.Int1();
sl@0
   706
	const TBool omitBlobs=aMessage.Int2();
sl@0
   707
sl@0
   708
	if (!omitBlobs && pRow)
sl@0
   709
		{	// write straight into cursor row
sl@0
   710
		ReadRowL(*pRow,size,aMessage);
sl@0
   711
		return;
sl@0
   712
		}
sl@0
   713
	RDbRow row;
sl@0
   714
	row.PushL();
sl@0
   715
	ReadRowL(row,size,aMessage);
sl@0
   716
	TInt max=aCursor.ColumnCount();
sl@0
   717
	for (TInt ii=0;++ii<=max;)
sl@0
   718
		{
sl@0
   719
		if (omitBlobs && TDbCol::IsLong(aCursor.ColumnType(ii)))
sl@0
   720
			;
sl@0
   721
		else
sl@0
   722
			aCursor.Column(ii).SetL(row.ColCell(ii));
sl@0
   723
		}
sl@0
   724
	CleanupStack::PopAndDestroy();		// row
sl@0
   725
	}
sl@0
   726
sl@0
   727
// Writes the row buffer to the client row buffer, if big enough
sl@0
   728
// If the client buffer is too small it will attempt a second retrieval
sl@0
   729
TInt CDbsSession::WriteRowL(const RDbRow& aRow,const RMessage2& aMessage)
sl@0
   730
	{
sl@0
   731
	TInt size=aRow.Size();
sl@0
   732
	if (size>0 && size<=aMessage.Int2())
sl@0
   733
		aMessage.WriteL(3,TPtrC8((const TUint8*)aRow.First(),size));
sl@0
   734
	return size+1;		// return 0 reserved for Goto
sl@0
   735
	}
sl@0
   736
sl@0
   737
// Returns a row to the client
sl@0
   738
TInt CDbsSession::RetrieveRowL(CDbCursor& aCursor,const RMessage2& aMessage)
sl@0
   739
	{
sl@0
   740
	RDbRow* pRow=aCursor.RowBuffer();
sl@0
   741
	if (pRow)	// direct row transfer
sl@0
   742
		return WriteRowL(*pRow,aMessage);
sl@0
   743
	// need to retrieve columns independantly
sl@0
   744
	RDbRow row;
sl@0
   745
	row.PushL();
sl@0
   746
	TInt max=aCursor.ColumnCount();
sl@0
   747
	for (TInt ii=0;++ii<=max;)
sl@0
   748
		TDbColumn(row,ii).SetL(aCursor.ColumnC(ii));
sl@0
   749
	TInt size=WriteRowL(row,aMessage);
sl@0
   750
	CleanupStack::PopAndDestroy();	// row
sl@0
   751
	return size;
sl@0
   752
	}
sl@0
   753
sl@0
   754
//This method creates new EDbsCursor type object.
sl@0
   755
//It is used at table level so it needs table object security policies.
sl@0
   756
//The related table MPolicy interface will be 
sl@0
   757
//put together with the EDbsCursor object in TEntry list.
sl@0
   758
//
sl@0
   759
//Complete the cursor construction
sl@0
   760
TInt CDbsSession::NewCursorL(CDbCursor* aCursor,const RMessage2& aMessage, const MPolicy* aTblSecurityPolicy)
sl@0
   761
	{
sl@0
   762
	aCursor->PushL();		// it has a context: use safe cleanup
sl@0
   763
	TPckgBuf<TDbsColumns> cols;
sl@0
   764
	TInt ii=cols().iCount=aCursor->ColumnCount();
sl@0
   765
	if (ii<=cols().EMax)
sl@0
   766
		{
sl@0
   767
		while (--ii>=0)
sl@0
   768
			cols().iData[ii]=TUint8(aCursor->ColumnType(ii+1));
sl@0
   769
		}
sl@0
   770
	aMessage.WriteL(3,cols);
sl@0
   771
	CleanupStack::Pop(aCursor);
sl@0
   772
	return Add(aCursor, aTblSecurityPolicy);
sl@0
   773
	}
sl@0
   774
sl@0
   775
//This method creates new EDbsIncremental type object.
sl@0
   776
//It is used either at database or  table level so it needs database or table object
sl@0
   777
//security policies. The related database/table MPolicy interface will be
sl@0
   778
//put together with the EDbsIncremental object in TEntry list.
sl@0
   779
//
sl@0
   780
//Complete a message which returns an incremental object by writing the step back
sl@0
   781
//When there is no work to do, aIncremental may be 0: then return a null handle
sl@0
   782
TInt CDbsSession::NewIncrementalL(CDbIncremental* aIncremental,
sl@0
   783
								  TInt& aInit,const RMessage2& aMessage,
sl@0
   784
								  const MPolicy* aPolicy)
sl@0
   785
	{
sl@0
   786
	aIncremental->PushL();		// it has a context: use safe cleanup
sl@0
   787
	aMessage.WriteL(3,TPckgC<TInt>(aInit));
sl@0
   788
	CleanupStack::Pop();
sl@0
   789
	return aIncremental ? Add(aIncremental, aPolicy) : 0;
sl@0
   790
	}
sl@0
   791
sl@0
   792
//This method creates new EDbsStream type object.
sl@0
   793
//It is used either at database or table level so it needs either 
sl@0
   794
//database or table object security policies.
sl@0
   795
//The related database/table MPolicy interface will be
sl@0
   796
//put together with the EDbsStream object in TEntry list.
sl@0
   797
//
sl@0
   798
//Complete a stream based message return
sl@0
   799
TInt CDbsSession::NewStreamL(MStreamBuf* aHost,const RMessage2& aMessage,
sl@0
   800
							 const MPolicy* aPolicy,TInt aExtent)
sl@0
   801
	{
sl@0
   802
	aHost->PushL();
sl@0
   803
	TPckgBuf<TDbsStreamBuf> buf;
sl@0
   804
	if (aExtent>0)	// read the first buffer-full
sl@0
   805
		aHost->ReadL(buf().iData,Min(aExtent,KDbsStreamBufSize));
sl@0
   806
	TInt h=0;
sl@0
   807
	if (aExtent<0 || aExtent>KDbsStreamBufSize)
sl@0
   808
		{	// create the stream object
sl@0
   809
		HDbsStream* stream = new(ELeave) HDbsStream(aHost,KDbsStreamBufSize);
sl@0
   810
		h=Add(stream, aPolicy);
sl@0
   811
		CleanupStack::Pop();		// aHost
sl@0
   812
		}
sl@0
   813
	else	// no more data to send
sl@0
   814
		CleanupStack::PopAndDestroy();		// aHost
sl@0
   815
	if (aExtent>=0)
sl@0
   816
		{
sl@0
   817
		buf().iExt=aExtent;
sl@0
   818
		aMessage.WriteL(3,buf);
sl@0
   819
		}
sl@0
   820
	return h;
sl@0
   821
	}
sl@0
   822
sl@0
   823
//This method creates new EDbsStream type object.
sl@0
   824
//It is used at database level so it needs the database object security policies.
sl@0
   825
//The related database MPolicy interface will be 
sl@0
   826
//put together with the EDbsStream object in TEntry list.
sl@0
   827
//
sl@0
   828
//Generic object passing code
sl@0
   829
//aPtr should be on the cleanup stack, aExter can externalize it
sl@0
   830
TInt CDbsSession::NewStreamL(TAny* aPtr,TExternalizeFunction aExter,
sl@0
   831
							 const RMessage2& aMessage, const MPolicy* aDbSecurityPolicy)
sl@0
   832
	{
sl@0
   833
	RWriteStream strm(HBufBuf::NewLC());
sl@0
   834
	aExter(aPtr,strm);
sl@0
   835
	strm.CommitL();
sl@0
   836
	TInt ext=strm.Sink()->SizeL();
sl@0
   837
	CleanupStack::Pop();			// host
sl@0
   838
	CleanupStack::PopAndDestroy();	// aPtr
sl@0
   839
	TInt res = NewStreamL(strm.Sink(),aMessage,aDbSecurityPolicy,ext);
sl@0
   840
	return res;
sl@0
   841
	}
sl@0
   842
sl@0
   843
//allocates a free entry if required
sl@0
   844
void CDbsSession::AllocL()
sl@0
   845
	{
sl@0
   846
	__ASSERT(TUint(iFree)<=TUint(iSize));
sl@0
   847
	if (iFree==iSize)
sl@0
   848
		{
sl@0
   849
		TInt size=iSize+EIndexGranularity;
sl@0
   850
		if (size>KDbsIndexLimit)	// maximum number of objects reached
sl@0
   851
			__LEAVE(KErrNoMemory);
sl@0
   852
		TEntry* ix=iIx=(TEntry*)User::ReAllocL(iIx,size*sizeof(TEntry));
sl@0
   853
		iSize=size;
sl@0
   854
		for (TInt ii=iFree;ii<size;)
sl@0
   855
			{
sl@0
   856
			TEntry& e=ix[ii];
sl@0
   857
			e.iNext=++ii;
sl@0
   858
			e.iType=TUint8(EDbsFree);
sl@0
   859
			e.iPolicy = NULL;
sl@0
   860
			}
sl@0
   861
		}
sl@0
   862
	}
sl@0
   863
sl@0
   864
// Sets the free entry and return a handle to it
sl@0
   865
TInt CDbsSession::DoAdd(TAny* aObject,TDbsType aType, const MPolicy* aPolicy)
sl@0
   866
	{
sl@0
   867
	__ASSERT(TUint(iFree)<TUint(iSize));
sl@0
   868
	__ASSERT(aType!=EDbsFree);
sl@0
   869
	TInt ix=iFree;
sl@0
   870
	TEntry& e=iIx[ix];
sl@0
   871
	__ASSERT(e.Type()==EDbsFree);
sl@0
   872
	iFree=e.iNext;
sl@0
   873
	__ASSERT(TUint(iFree)<=TUint(iSize));
sl@0
   874
	e.iObject=aObject;
sl@0
   875
	e.iType=TUint8(aType);
sl@0
   876
	e.iPolicy = aPolicy;
sl@0
   877
	TInt magic=(e.iMagic+1)&KDbsMagicMask;
sl@0
   878
	e.iMagic=TUint8(magic);
sl@0
   879
	return DbsMakeHandle(ix,magic,aType);
sl@0
   880
	}
sl@0
   881
sl@0
   882
// releases the object and the entry to the free pool
sl@0
   883
void CDbsSession::Free(TEntry& aEntry)
sl@0
   884
	{
sl@0
   885
	__ASSERT(TUint(iFree)<=TUint(iSize));
sl@0
   886
	aEntry.Release();
sl@0
   887
	aEntry.iNext=iFree;
sl@0
   888
	iFree=&aEntry-&iIx[0];
sl@0
   889
	__ASSERT(TUint(iFree)<TUint(iSize));
sl@0
   890
	}
sl@0
   891
sl@0
   892
CDbsSession::TEntry& CDbsSession::Object(TInt aHandle)
sl@0
   893
	{
sl@0
   894
	TEntry* e=0;
sl@0
   895
	TInt ix=DbsObjectIndex(aHandle);
sl@0
   896
	if (TUint(ix)<TUint(iSize))
sl@0
   897
		{
sl@0
   898
		e=&iIx[ix];
sl@0
   899
		if (DbsMakeHandle(ix,e->iMagic,e->Type())!=aHandle)
sl@0
   900
			e=0;
sl@0
   901
		}
sl@0
   902
	__ASSERT_ALWAYS(e && e->iPolicy,::Panic(EDbsBadHandle));
sl@0
   903
	return *e;
sl@0
   904
	}
sl@0
   905
sl@0
   906
// Reports how many objects are allocated by the client
sl@0
   907
TInt CDbsSession::CountResources()
sl@0
   908
	{
sl@0
   909
	TInt alloc=0;
sl@0
   910
	const TEntry* const base=iIx;
sl@0
   911
	if (base)
sl@0
   912
		{
sl@0
   913
		for (const TEntry* e=base+iSize;--e>=base;)
sl@0
   914
			{
sl@0
   915
			if (e->Type()!=EDbsFree)
sl@0
   916
				++alloc;
sl@0
   917
			}
sl@0
   918
		}
sl@0
   919
	return alloc;
sl@0
   920
	}
sl@0
   921
sl@0
   922
#ifdef __DBDUMP__
sl@0
   923
//Using CDbsSession::Dump() method you can dump the session content
sl@0
   924
//into a stream. Note that the dump works only if you have __DBDUMP__ macro defined.
sl@0
   925
void CDbsSession::Dump()
sl@0
   926
	{
sl@0
   927
	RFile dumpFile;
sl@0
   928
	RFs fileSess;
sl@0
   929
	TInt err = fileSess.Connect();
sl@0
   930
	if(err == KErrNone)
sl@0
   931
		{
sl@0
   932
		_LIT(KDumpFileName, "_:\\PUBLIC\\DBMS\\DUMP%X.TXT");
sl@0
   933
		TFileName dumpFileName(KDumpFileName);
sl@0
   934
		dumpFileName[0] = 'A' + static_cast<TInt>(RFs::GetSystemDrive());
sl@0
   935
		
sl@0
   936
		TBuf<40> buf;
sl@0
   937
		buf.Format(dumpFileName, this);
sl@0
   938
		err = fileSess.MkDirAll(buf);
sl@0
   939
		if(err == KErrNone || err == KErrAlreadyExists)
sl@0
   940
			{
sl@0
   941
			err = dumpFile.Replace(fileSess, buf, EFileWrite);
sl@0
   942
			if(err == KErrNone)
sl@0
   943
				{
sl@0
   944
				TEntry* const base = iIx;
sl@0
   945
				if(base)
sl@0
   946
					{
sl@0
   947
					for(TEntry* e=base+iSize;--e>=base;)
sl@0
   948
						{
sl@0
   949
						if(e->Type() != EDbsFree)
sl@0
   950
							{
sl@0
   951
							e->Dump(dumpFile);
sl@0
   952
							}
sl@0
   953
						}
sl@0
   954
					}//end of - if(base)
sl@0
   955
				}//end of - if(err == KErrNone)
sl@0
   956
			}
sl@0
   957
		}//end of - if(err == KErrNone)
sl@0
   958
	dumpFile.Close();
sl@0
   959
	fileSess.Close();
sl@0
   960
	RDebug::Print(_L("CDbsSession::Dump() error=%d\n"), err);
sl@0
   961
	__ASSERT_ALWAYS(err == KErrNone, User::Invariant());
sl@0
   962
	}
sl@0
   963
#endif//__DBDUMP__
sl@0
   964
sl@0
   965
/**
sl@0
   966
Reserves a prederfined amount of disk space on aDrive drive. 
sl@0
   967
At the moment the max possible amount, allowed by the file server, is reserved on aDrive drive.
sl@0
   968
sl@0
   969
Use this call to ensure that if your "delete records" transaction fails because of "out of
sl@0
   970
disk space" circumstances, you can get an access to the already reserved disk space and
sl@0
   971
complete your transaction successfully the second time.
sl@0
   972
sl@0
   973
There is no strong, 100% guarantee, that the reserved disk space will always help the client
sl@0
   974
in "low memory" situations.
sl@0
   975
sl@0
   976
This method processes EDbsReserveDriveSpace message.
sl@0
   977
sl@0
   978
@param aDrive Drive number to reserve space on
sl@0
   979
@leave KErrArgument Invalid aDrive value.
sl@0
   980
@leave CDbsSessDriveSpace::ReserveL() leaving error codes
sl@0
   981
@see CDbsSessDriveSpace::ReserveL()
sl@0
   982
*/
sl@0
   983
void CDbsSession::ReserveDriveSpaceL(TDriveNumber aDrive)
sl@0
   984
	{
sl@0
   985
	if(aDrive < EDriveA || aDrive > EDriveZ)
sl@0
   986
        {
sl@0
   987
        __LEAVE(KErrArgument);
sl@0
   988
        }
sl@0
   989
    iSessDriveSpace->ReserveL(aDrive);
sl@0
   990
	}
sl@0
   991
sl@0
   992
/**
sl@0
   993
The method frees the reserved by the DBMS session disk space.
sl@0
   994
sl@0
   995
This method processes EDbsFreeReservedSpace message.
sl@0
   996
sl@0
   997
@param aDrive Drive number, which reserved space has to be freed.
sl@0
   998
@see CDbsSessDriveSpace::Free()
sl@0
   999
@panic Client side panic "DBMS-server 10" in debug mode, if the drive number is invalid.
sl@0
  1000
@panic In debug mode there will be a panic with the line number as an error code if 
sl@0
  1001
       there is no reserved disk space for aDrive. 
sl@0
  1002
@panic In debug mode there will be a panic with the line number as an error code if 
sl@0
  1003
       the reserved disk space is granted but not released.
sl@0
  1004
*/
sl@0
  1005
void CDbsSession::FreeReservedSpace(TDriveNumber aDrive)
sl@0
  1006
    {
sl@0
  1007
    TBool valid = aDrive >= EDriveA && aDrive <= EDriveZ;
sl@0
  1008
    __ASSERT_DEBUG(valid, ::Panic(EDbsInvalidDrive));
sl@0
  1009
    if(valid)
sl@0
  1010
        {
sl@0
  1011
        iSessDriveSpace->Free(aDrive);
sl@0
  1012
        }
sl@0
  1013
    }
sl@0
  1014
sl@0
  1015
/**
sl@0
  1016
Grants access to a given area on a given drive for CDbsSession session.
sl@0
  1017
Note this session must have reserved space on this particular drive in order to be 
sl@0
  1018
granted access to the reserved area.
sl@0
  1019
sl@0
  1020
This method processes EDbsReserveGetAccess message.
sl@0
  1021
  
sl@0
  1022
@param aDrive Drive number with a reserved disk space, an access to which is requested.
sl@0
  1023
@leave KErrArgument Invalid drive.
sl@0
  1024
@leave CDbsSessDriveSpace::GrantAccessL() leaving error codes
sl@0
  1025
@see CDbsSessDriveSpace::GrantAccessL()
sl@0
  1026
*/
sl@0
  1027
void CDbsSession::GetReserveAccessL(TDriveNumber aDrive)
sl@0
  1028
	{
sl@0
  1029
	if(aDrive < EDriveA || aDrive > EDriveZ)
sl@0
  1030
        {
sl@0
  1031
        __LEAVE(KErrArgument);
sl@0
  1032
        }
sl@0
  1033
    iSessDriveSpace->GrantAccessL(aDrive);
sl@0
  1034
	}
sl@0
  1035
sl@0
  1036
/**
sl@0
  1037
Revokes access on a given drive for CDbsSession session.
sl@0
  1038
sl@0
  1039
This method processes EDbsReserveReleaseAccess message.
sl@0
  1040
  
sl@0
  1041
@param aDrive Drive number with a reserved disk space, the access to which has to be released.
sl@0
  1042
@panic Client side panic "DBMS-server 10" in debug mode, if the drive number is invalid.
sl@0
  1043
@panic In debug mode there will be a panic with the line number as an error code if 
sl@0
  1044
       there is no reserved disk space for aDrive. 
sl@0
  1045
@panic In debug mode there will be a panic with the line number as an error code if 
sl@0
  1046
       there is no granted access to the reserved disk space. 
sl@0
  1047
*/
sl@0
  1048
void CDbsSession::ReleaseReserveAccess(TDriveNumber aDrive)
sl@0
  1049
	{
sl@0
  1050
    TBool valid = aDrive >= EDriveA && aDrive <= EDriveZ;
sl@0
  1051
    __ASSERT_DEBUG(valid, ::Panic(EDbsInvalidDrive));
sl@0
  1052
    if(valid)
sl@0
  1053
        {
sl@0
  1054
        iSessDriveSpace->ReleaseAccess(aDrive);
sl@0
  1055
        }
sl@0
  1056
	}