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