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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // DBMS server-session and support classes
20 #include "Sd_DbList.h"
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.
68 // Class CDbsSession::TEntry
73 void CDbsSession::TEntry::Release()
76 __ASSERT(t!=EDbsFree);
78 delete STATIC_CAST(CDbsObserver::HObserver*,iObject);
79 else if (t==EDbsStream)
80 delete STATIC_CAST(HDbsStream*,iObject);
82 CDbObject::Destroy(STATIC_CAST(CDbObject*,iObject));
83 iType=TUint8(EDbsFree);
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
92 _LIT8(KClassName, "CDbsSession::TEntry. this=%X");
93 _LIT8(KType, "Type=%S");
94 _LIT8(KMagic, "Magic=%d");
97 TPtrC8 KDbsType[EDbsMaxType + 1] = {_L8("Free"), _L8("Database"), _L8("Incremental"), _L8("Cursor"), _L8("Constraint"), _L8("Stream"), _L8("Observer")};
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);
110 iPolicy->Dump(aFile);
111 (void)aFile.Write(KEnd);
112 (void)aFile.Write(KCrLf);
119 // Release all this clients resources
120 // Streams must be released before cursors, and everything before the connection
122 CDbsSession::~CDbsSession()
124 TEntry* const base=&iIx[0];
127 TInt type=EDbsMaxType;
129 for (TEntry* e=base+iSize;--e>=base;)
134 } while (--type>EDbsFree);
137 iDbPolicyRqColl.Close();
138 delete iSessDriveSpace;
139 Server().RemoveSession();
143 Overrides virtual CSession2::Create2().
144 Creates iSessDriveSpace instance.
146 void CDbsSession::CreateL()
148 iSessDriveSpace = CDbsSessDriveSpace::NewL(Server().DriveSpaceCol());
152 // provide for CSession
153 // If this leaves, we complete message through ServiceError(...)
155 void CDbsSession::ServiceL(const RMessage2& aMessage)
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)
168 case EDbsResourceMark:
169 ResourceCountMarkStart();
171 case EDbsResourceCheck:
172 ResourceCountMarkEnd(aMessage);
174 case EDbsResourceCount:
177 case EDbsSetHeapFailure:
178 User::__DbgSetAllocFail(RHeap::EUser,RHeap::TAllocFail(aMessage.Int0()),aMessage.Int1());
180 case EDbsOpenDatabase:
181 r=OpenDatabaseL(aMessage);
183 case EDbsReserveDriveSpace:
184 ReserveDriveSpaceL(static_cast <TDriveNumber> (aMessage.Int0()));
186 case EDbsFreeReservedSpace:
187 FreeReservedSpace(static_cast <TDriveNumber> (aMessage.Int0()));
189 case EDbsReserveGetAccess:
190 GetReserveAccessL(static_cast <TDriveNumber> (aMessage.Int0()));
192 case EDbsReserveReleaseAccess:
193 ReleaseReserveAccess(static_cast <TDriveNumber> (aMessage.Int0()));
196 r=ExtServiceL(aMessage, static_cast <TDbsFunction> (f&~KDbsObjectReturn));
201 { // object based requests
204 policyProxy.CheckL(aMessage, *e.iPolicy);
205 TDbsFunction dbmsFuncNo = static_cast <TDbsFunction> (f & ~KDbsObjectReturn);
208 // the common close function
210 //If this is a database object, remove the related pair (handle, uid) from iDbPolicyRqColl collection.
211 if(e.Type() == EDbsDatabase)
213 iDbPolicyRqColl.Remove(h);
217 // database functions
218 case EDbsDatabaseAuthenticate:
220 __LEAVE(KErrNotSupported);
223 case EDbsDatabaseDestroy:
224 r=e.Database().Destroy();
226 case EDbsDatabaseBegin:
227 r=e.Database().Begin();
229 case EDbsDatabaseCommit:
230 r=e.Database().Commit();
232 case EDbsDatabaseRollback:
233 e.Database().Rollback();
235 case EDbsDatabaseProperty:
236 r=e.Database().Property(CDbDatabase::TProperty(aMessage.Int0()));
238 case EDbsDatabaseTables:
239 {//The new stream object will have the same security policy as the database
241 CDbTableNames* names=CDbTableNames::NewLC();
242 e.Database().TablesL(*names);
243 r=NewStreamL(names,Externalizer(names),aMessage,e.iPolicy);
246 case EDbsDatabaseColumns:
247 {//The new stream object will have the same security policy as the database
249 CDbColSet* set=CDbColSet::NewLC();
250 e.Database().ColumnsL(*set,ReadName0L(0,aMessage));
251 r=NewStreamL(set,Externalizer(set),aMessage,e.iPolicy);
254 case EDbsDatabaseIndexes:
255 {//The new stream object will have the same security policy as the database
257 CDbIndexNames* names=CDbIndexNames::NewLC();
258 e.Database().IndexesL(*names,ReadName0L(0,aMessage));
259 r=NewStreamL(names,Externalizer(names),aMessage,e.iPolicy);
262 case EDbsDatabaseKeys:
263 {//The new stream object will have the same security policy as the database
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);
270 case EDbsDatabaseCreateTable:
272 const TDesC& name=ReadName0L(0,aMessage);
273 CDbColSet* set=ColSetLC(1,aMessage);
276 primary=KeyLC(2,aMessage);
278 e.Database().CreateTableL(name,*set,primary);
280 CleanupStack::PopAndDestroy(); // primary
281 CleanupStack::PopAndDestroy(); // set
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);
288 case EDbsDatabaseOpenUtility:
289 {//The new incremental object will have the same security policy as the database
291 r=NewIncrementalL(e.Database().UtilityL(CDbDatabase::TUtility(aMessage.Int0()),step),step,aMessage,e.iPolicy);
294 case EDbsDatabaseOpenDropTable:
295 {//The new incremental object will have the same security policy as the database
297 const TDesC& name=ReadName0L(0,aMessage);
298 r=NewIncrementalL(e.Database().DropTableL(name,step),step,aMessage,e.iPolicy);
301 case EDbsDatabaseOpenAlterTable:
302 {//The new incremental object will have the same security policy as the database
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
310 case EDbsDatabaseOpenCreateIndex:
311 {//The new incremental object will have the same security policy as the database
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
320 case EDbsDatabaseOpenDropIndex:
321 {//The new incremental object will have the same security policy as the database
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);
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.
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);
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);
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);
373 r=NewCursorL(e.Database().TableL(name,accessType),aMessage,policy);
376 // Incremental functions
377 case EDbsIncrementalNext:
379 TPckgBuf<TInt> step=aMessage.Int0();
380 r=e.Incremental().NextL(step());
381 aMessage.WriteL(1,step);
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();
392 case EDbsCursorColumnTypes:
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();
402 case EDbsCursorReset:
405 case EDbsCursorEvaluate:
406 r=e.Cursor().EvaluateL()?1:0;
408 case EDbsCursorUnevaluated:
409 r=e.Cursor().Unevaluated()?1:0;
411 case EDbsCursorSetIndex:
415 name=&ReadName0L(0,aMessage);
416 e.Cursor().SetIndexL(name);
421 TDbLookupKey* key=LookupKeyLC(0,aMessage.Int1(),aMessage);
422 r=e.Cursor().SeekL(*key,RDbTable::TComparison(aMessage.Int2()))?1:0;
423 CleanupStack::PopAndDestroy(); // key;
426 case EDbsCursorAtBeginning:
427 r=e.Cursor().AtBeginning()?1:0;
429 case EDbsCursorAtEnd:
430 r=e.Cursor().AtEnd()?1:0;
432 case EDbsCursorAtRow:
433 r=e.Cursor().AtRow()?1:0;
435 case EDbsCursorCount:
436 r=e.Cursor().CountL(RDbRowSet::TAccuracy(aMessage.Int0()))+1;
438 case EDbsCursorGotoPos:
439 if (e.Cursor().GotoL(RDbRowSet::TPosition(aMessage.Int0())))
442 r=RetrieveRowL(e.Cursor(),aMessage);
445 case EDbsCursorBookmark:
447 TPckgBuf<TDbBookmark::TMark> mark;
448 e.Cursor().Bookmark(mark());
449 aMessage.WriteL(3,mark);
452 case EDbsCursorGotoBookmark:
454 TPckgBuf<TDbBookmark::TMark> mark;
455 aMessage.ReadL(0,mark);
456 e.Cursor().GotoL(mark());
461 r=RetrieveRowL(e.Cursor(),aMessage);
463 case EDbsCursorInsert:
464 e.Cursor().InsertL(CDbCursor::TInsert(aMessage.Int0()));
465 r=RetrieveRowL(e.Cursor(),aMessage);
467 case EDbsCursorUpdate:
468 e.Cursor().UpdateL();
469 r=RetrieveRowL(e.Cursor(),aMessage);
471 case EDbsCursorRetrieveRow: // pass the row buffer back to the client
472 r=RetrieveRowL(e.Cursor(),aMessage);
474 case EDbsCursorCancel:
478 PutRowL(e.Cursor(),aMessage);
481 case EDbsCursorDelete:
482 e.Cursor().DeleteL();
484 case EDbsCursorColumns:
485 {// reduce memory usage by extracting and stream columns individually
487 RWriteStream strm(HBufBuf::NewLC());
488 TInt count=e.Cursor().ColumnCount();
489 strm.WriteInt32L(count);
491 for (TInt ii=0;++ii<=count;)
493 e.Cursor().ColumnDef(col,ii);
497 TInt ext=strm.Sink()->SizeL();
498 CleanupStack::Pop(); // stream
499 r=NewStreamL(strm.Sink(),aMessage,e.iPolicy,ext);
502 case EDbsCursorColumnDef:
504 TPckgBuf<TDbCol> col;
505 e.Cursor().ColumnDef(col(),aMessage.Int0());
506 aMessage.WriteL(3,col);
509 case EDbsCursorSetNull:
510 __ASSERT(TDbCol::IsLong(e.Cursor().ColumnType(aMessage.Int0())));
511 e.Cursor().SetNullL(aMessage.Int0());
513 case EDbsCursorColumnSize:
514 __ASSERT(TDbCol::IsLong(e.Cursor().ColumnType(aMessage.Int0())));
515 r=e.Cursor().ColumnSize(aMessage.Int0());
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));
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);
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);
539 case EDbsCursorMatch:
540 r=e.Cursor().MatchL(Object(aMessage.Int0()).Constraint())?1:0;
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
553 r=e.Stream().ReadL(aMessage);
556 case EDbsStreamWrite:
558 e.Stream().WriteL(aMessage);
563 r=e.Stream().SizeL();
566 case EDbsStreamSynch:
571 // we don't know about any other functions
578 // don't complete message if we have paniced client, just reset variable;
579 if (!aMessage.IsNull())
580 aMessage.Complete(r);
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)
592 if (aError==KErrBadDescriptor)
594 // this completes message as well so we have to handle it in
596 _LIT(KCategory,"DBMS-server");
597 Server().Panic(KCategory,EDbsBadDescriptor);
599 CSession2::ServiceError(aMessage,aError);
602 // Read aPtr as a name
603 const TDesC& CDbsSession::ReadName0L(TInt aIndex,const RMessage2& aMessage)
605 TDes& name=Server().Name0();
606 aMessage.ReadL(aIndex,name);
610 // Read Ptr1 as a name
611 const TDesC& CDbsSession::ReadName1L(TInt aIndex,const RMessage2& aMessage)
613 TDes& name=Server().Name1();
614 aMessage.ReadL(aIndex,name);
618 // Reads aPtr as a variable length 8-bit descriptor
619 HBufC8* CDbsSession::ReadHBuf8LC(TInt aIndex,const RMessage2& aMessage)
621 TInt len= aMessage.GetDesLengthL(aIndex);
622 HBufC8* buf=HBufC8::NewLC(len);
623 TPtr8 read(buf->Des());
624 aMessage.ReadL(aIndex,read);
628 // Reads aPtr as a variable length build-width descriptor
629 HBufC* CDbsSession::ReadHBufLC(TInt aIndex,const RMessage2& aMessage)
631 TInt len= aMessage.GetDesLengthL(aIndex);
632 HBufC* buf=HBufC::NewLC(len);
633 TPtr read(buf->Des());
634 aMessage.ReadL(aIndex,read);
638 // Pulls a column set from the client
639 CDbColSet* CDbsSession::ColSetLC(TInt aIndex,const RMessage2& aMessage)
641 CDbColSet* set=CDbColSet::NewLC();
642 HBufC8* data=ReadHBuf8LC(aIndex,aMessage);
643 RDesReadStream strm(*data);
645 CleanupStack::PopAndDestroy(); // data
649 // Pulls a key from the client
650 CDbKey* CDbsSession::KeyLC(TInt aIndex,const RMessage2& aMessage)
652 CDbKey* key=CDbKey::NewLC();
653 HBufC8* data=ReadHBuf8LC(aIndex,aMessage);
654 RDesReadStream strm(*data);
656 CleanupStack::PopAndDestroy(); // data
660 // Reconstructs a TDbLookupKey from the message parameter
661 TDbLookupKey* CDbsSession::LookupKeyLC(TInt aIndex,TInt aSize,const RMessage2& aMessage)
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...
677 iter->iDes8.iPtr=(TUint8*)key+TInt(iter->iDes8.iPtr);
680 iter->iDes16.iPtr=(TUint16*)((TUint8*)key+TInt(iter->iDes8.iPtr));
683 } while (++iter<end);
688 // Reads the row from the client buffer into the server-side buffer
689 void CDbsSession::ReadRowL(RDbRow& aRow,TInt aSize,const RMessage2& aMessage)
694 TPtr8 buf((TUint8*)aRow.First(),aSize);
695 aMessage.ReadL(0,buf);
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)
704 RDbRow* pRow=aCursor.RowBuffer();
705 TInt size= aMessage.Int1();
706 const TBool omitBlobs=aMessage.Int2();
708 if (!omitBlobs && pRow)
709 { // write straight into cursor row
710 ReadRowL(*pRow,size,aMessage);
715 ReadRowL(row,size,aMessage);
716 TInt max=aCursor.ColumnCount();
717 for (TInt ii=0;++ii<=max;)
719 if (omitBlobs && TDbCol::IsLong(aCursor.ColumnType(ii)))
722 aCursor.Column(ii).SetL(row.ColCell(ii));
724 CleanupStack::PopAndDestroy(); // row
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)
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
737 // Returns a row to the client
738 TInt CDbsSession::RetrieveRowL(CDbCursor& aCursor,const RMessage2& aMessage)
740 RDbRow* pRow=aCursor.RowBuffer();
741 if (pRow) // direct row transfer
742 return WriteRowL(*pRow,aMessage);
743 // need to retrieve columns independantly
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
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.
759 //Complete the cursor construction
760 TInt CDbsSession::NewCursorL(CDbCursor* aCursor,const RMessage2& aMessage, const MPolicy* aTblSecurityPolicy)
762 aCursor->PushL(); // it has a context: use safe cleanup
763 TPckgBuf<TDbsColumns> cols;
764 TInt ii=cols().iCount=aCursor->ColumnCount();
768 cols().iData[ii]=TUint8(aCursor->ColumnType(ii+1));
770 aMessage.WriteL(3,cols);
771 CleanupStack::Pop(aCursor);
772 return Add(aCursor, aTblSecurityPolicy);
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.
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)
786 aIncremental->PushL(); // it has a context: use safe cleanup
787 aMessage.WriteL(3,TPckgC<TInt>(aInit));
789 return aIncremental ? Add(aIncremental, aPolicy) : 0;
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.
798 //Complete a stream based message return
799 TInt CDbsSession::NewStreamL(MStreamBuf* aHost,const RMessage2& aMessage,
800 const MPolicy* aPolicy,TInt aExtent)
803 TPckgBuf<TDbsStreamBuf> buf;
804 if (aExtent>0) // read the first buffer-full
805 aHost->ReadL(buf().iData,Min(aExtent,KDbsStreamBufSize));
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
813 else // no more data to send
814 CleanupStack::PopAndDestroy(); // aHost
818 aMessage.WriteL(3,buf);
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.
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)
833 RWriteStream strm(HBufBuf::NewLC());
836 TInt ext=strm.Sink()->SizeL();
837 CleanupStack::Pop(); // host
838 CleanupStack::PopAndDestroy(); // aPtr
839 TInt res = NewStreamL(strm.Sink(),aMessage,aDbSecurityPolicy,ext);
843 //allocates a free entry if required
844 void CDbsSession::AllocL()
846 __ASSERT(TUint(iFree)<=TUint(iSize));
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));
854 for (TInt ii=iFree;ii<size;)
858 e.iType=TUint8(EDbsFree);
864 // Sets the free entry and return a handle to it
865 TInt CDbsSession::DoAdd(TAny* aObject,TDbsType aType, const MPolicy* aPolicy)
867 __ASSERT(TUint(iFree)<TUint(iSize));
868 __ASSERT(aType!=EDbsFree);
871 __ASSERT(e.Type()==EDbsFree);
873 __ASSERT(TUint(iFree)<=TUint(iSize));
875 e.iType=TUint8(aType);
877 TInt magic=(e.iMagic+1)&KDbsMagicMask;
878 e.iMagic=TUint8(magic);
879 return DbsMakeHandle(ix,magic,aType);
882 // releases the object and the entry to the free pool
883 void CDbsSession::Free(TEntry& aEntry)
885 __ASSERT(TUint(iFree)<=TUint(iSize));
888 iFree=&aEntry-&iIx[0];
889 __ASSERT(TUint(iFree)<TUint(iSize));
892 CDbsSession::TEntry& CDbsSession::Object(TInt aHandle)
895 TInt ix=DbsObjectIndex(aHandle);
896 if (TUint(ix)<TUint(iSize))
899 if (DbsMakeHandle(ix,e->iMagic,e->Type())!=aHandle)
902 __ASSERT_ALWAYS(e && e->iPolicy,::Panic(EDbsBadHandle));
906 // Reports how many objects are allocated by the client
907 TInt CDbsSession::CountResources()
910 const TEntry* const base=iIx;
913 for (const TEntry* e=base+iSize;--e>=base;)
915 if (e->Type()!=EDbsFree)
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()
929 TInt err = fileSess.Connect();
932 _LIT(KDumpFileName, "_:\\PUBLIC\\DBMS\\DUMP%X.TXT");
933 TFileName dumpFileName(KDumpFileName);
934 dumpFileName[0] = 'A' + static_cast<TInt>(RFs::GetSystemDrive());
937 buf.Format(dumpFileName, this);
938 err = fileSess.MkDirAll(buf);
939 if(err == KErrNone || err == KErrAlreadyExists)
941 err = dumpFile.Replace(fileSess, buf, EFileWrite);
944 TEntry* const base = iIx;
947 for(TEntry* e=base+iSize;--e>=base;)
949 if(e->Type() != EDbsFree)
955 }//end of - if(err == KErrNone)
957 }//end of - if(err == KErrNone)
960 RDebug::Print(_L("CDbsSession::Dump() error=%d\n"), err);
961 __ASSERT_ALWAYS(err == KErrNone, User::Invariant());
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.
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.
973 There is no strong, 100% guarantee, that the reserved disk space will always help the client
974 in "low memory" situations.
976 This method processes EDbsReserveDriveSpace message.
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()
983 void CDbsSession::ReserveDriveSpaceL(TDriveNumber aDrive)
985 if(aDrive < EDriveA || aDrive > EDriveZ)
987 __LEAVE(KErrArgument);
989 iSessDriveSpace->ReserveL(aDrive);
993 The method frees the reserved by the DBMS session disk space.
995 This method processes EDbsFreeReservedSpace message.
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.
1005 void CDbsSession::FreeReservedSpace(TDriveNumber aDrive)
1007 TBool valid = aDrive >= EDriveA && aDrive <= EDriveZ;
1008 __ASSERT_DEBUG(valid, ::Panic(EDbsInvalidDrive));
1011 iSessDriveSpace->Free(aDrive);
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.
1020 This method processes EDbsReserveGetAccess message.
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()
1027 void CDbsSession::GetReserveAccessL(TDriveNumber aDrive)
1029 if(aDrive < EDriveA || aDrive > EDriveZ)
1031 __LEAVE(KErrArgument);
1033 iSessDriveSpace->GrantAccessL(aDrive);
1037 Revokes access on a given drive for CDbsSession session.
1039 This method processes EDbsReserveReleaseAccess message.
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.
1048 void CDbsSession::ReleaseReserveAccess(TDriveNumber aDrive)
1050 TBool valid = aDrive >= EDriveA && aDrive <= EDriveZ;
1051 __ASSERT_DEBUG(valid, ::Panic(EDbsInvalidDrive));
1054 iSessDriveSpace->ReleaseAccess(aDrive);