Update contrib.
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.
18 // Class RDbTransaction::CNotifier
20 NONSHARABLE_CLASS(RDbTransaction::CNotifier) : public CDbNotifier
23 inline CNotifier( RDbTransaction& aTransaction );
26 // void Event( RDbNotifier::TEvent aEvent );
28 // void Complete( TInt aStatus );
30 // void Notify( TType aEvent, TRequestStatus& aStatus );
33 RDbTransaction* iTransaction;
34 // TRequestStatus* iStatus;
38 inline RDbTransaction::CNotifier::CNotifier( RDbTransaction& aTransaction )
39 : iTransaction( &aTransaction )
42 RDbTransaction::CNotifier::~CNotifier()
44 // Cancel any outstanding request and extract from the transaction
50 __ASSERT( iTransaction->iNotifier == this );
51 iTransaction->iNotifier = 0;
55 //void RDbTransaction::CNotifier::Complete( TInt aStatus )
60 // User::RequestComplete( iStatus, aStatus );
64 //void RDbTransaction::CNotifier::Notify( CDbNotifier::TType aType, TRequestStatus& aStatus )
66 //// Request for future notification. If the database is closed complete immediately
69 // __ASSERT( !iStatus );
70 // __ASSERT( iPending >= 0 );
71 // iStatus = &aStatus;
72 // if ( iPending > RDbNotifier::EUnlock )
73 // Complete( iPending );
74 // else if ( !iTransaction )
75 // Complete( RDbNotifier::EClose );
79 // aStatus = KRequestPending;
83 void RDbTransaction::CNotifier::Cancel()
85 // Complete( KErrCancel );
88 //void RDbTransaction::CNotifier::Event( RDbNotifier::TEvent aEvent )
90 // if ( aEvent == RDbNotifier::EClose )
94 // __ASSERT( iPending < 0 );
95 // if (aEvent == RDbNotifier::EUnlock && iPending == CDbNotifier::EChange )
96 // ; // not interested in unlock events
98 // Complete( aEvent );
102 // __ASSERT( iPending >= 0 );
103 // if ( aEvent > iPending )
104 // iPending = aEvent; // save the event
109 // Class RDbTransaction
113 //void RDbTransaction::_Invariant() const
118 // if ( iLockCount == 0 )
119 // { // nothing must be happening in this state
120 // __ASSERT( iLockState == EDbReadLock );
121 // __ASSERT( iAction == EDbReadLock );
122 // __ASSERT( iUpdaters == 0 );
125 // switch ( iLockState & EState )
131 // __ASSERT( iAction == EDbReadLock );
132 // __ASSERT( iLockCount > 0 ); // someone must have a lock
133 // __ASSERT( iLockCount <= iMaxLock );
134 // __ASSERT( iUpdaters == 0 );
135 // __ASSERT( iPrimary.iState != 0 );
136 // for (TInt ii = iLockCount - 1; --ii >= 0; )
137 // __ASSERT( iSharers[ii].iState != 0 );
140 // case EDbCompactLock: // not allowed in user-transactions
141 // case EDbRecoveryLock:
142 // __ASSERT( iAction == iLockState );
143 // __ASSERT( iLockCount == 1 ); // exactly one lock allowed
144 // __ASSERT( iUpdaters == 0 );
145 // __ASSERT( iPrimary.iState == 0 );
147 // case EDbXReadLock: // intention to write. No updates but exclusive
148 // __ASSERT( iLockCount == 1 ); // exactly one lock allowed
149 // switch ( iAction )
153 // case EDbReadLock: // must be in a transaction: cannot commit a write/schema mod when releasing a read lock
154 // __ASSERT( iUpdaters == 0 );
155 // __ASSERT( iPrimary.iState & static_cast<TUint>( ETransactionLock ) );
157 // case EDbWriteLock:
158 // __ASSERT( iUpdaters > 0 );
162 // case EDbWriteLock:
163 // case EDbSchemaLock:
164 // __ASSERT( iLockCount == 1 ); // exactly one lock allowed
165 // switch ( iAction )
169 // case EDbReadLock: // must be in a transaction: cannot commit a write/schema mod when releasing a read lock
170 // __ASSERT( iUpdaters == 0 );
171 // __ASSERT( iPrimary.iState & static_cast<TUint>( ETransactionLock ) );
173 // case EDbWriteLock:
174 // __ASSERT( iUpdaters > 0 );
175 // __ASSERT( ( iLockState & EState ) == EDbWriteLock || ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) );
177 // case EDbSchemaLock:
178 // __ASSERT( ( iLockState & EState ) == EDbSchemaLock );
179 // __ASSERT( iUpdaters == 0 );
186 //template <class T> struct _InvariantFunc
188 // static void Invariant( TAny* aPtr ) { ( (const T*)aPtr )->_Invariant(); }
191 //template <class T> inline TCleanupOperation _InvariantFunction( T* )
192 // { return _InvariantFunc<T>::Invariant; }
196 // inline _Invariant( TCleanupOperation aOp, TAny* aPtr )
197 // : iOp( aOp ), iPtr( aPtr )
199 // inline ~_Invariant()
202 // TCleanupOperation iOp;
206 #ifndef __LEAVE_EQUALS_THROW
209 // inline _InvariantL( TCleanupOperation aOp, TAny* aPtr )
210 // { aOp( aPtr ); CleanupStack::PushL( TCleanupItem( aOp, aPtr ) ); }
211 // inline ~_InvariantL()
212 // { CleanupStack::PopAndDestroy(); }
214 #endif //__LEAVE_EQUALS_THROW__
216 //#define __INVARIANT struct _Invariant _invariant( _InvariantFunction( this ), this );
218 //#ifdef __LEAVE_EQUALS_THROW__
219 // #define __INVARIANT_L __INVARIANT
221 // #define __INVARIANT_L struct _InvariantL _invariant( _InvariantFunction( this ), this );
222 //#endif //__LEAVE_EQUALS_THROW__
224 #define __INVARIANT ( (void)0 )
225 #define __INVARIANT_L ( (void)0 )
228 #define __INVARIANT ( (void)0 )
229 #define __INVARIANT_L ( (void)0 )
231 #endif // _ASSERTIONS
233 inline TDbLockType RDbTransaction::LockState() const
234 { return TDbLockType( iLockState & EState ); }
236 void RDbTransaction::Close()
238 __ASSERT( !IsLocked() );
239 User::Free( iSharers );
240 // Event( RDbNotifier::EClose );
243 void RDbTransaction::DoCommitL()
245 // Commit any changes
248 __ASSERT_ALWAYS( ( iPrimary.iState & ~ETransactionLock ) == 0, Panic( EDbStreamsPendingOnCommit ) );
249 iLockState |= EFailed;
250 Database().FlushL( LockState() );
251 Database().SynchL( LockState() );
252 Unlock( RDbNotifier::ECommit );
255 void RDbTransaction::DoRollback()
257 // Rollback any changes
260 __ASSERT_ALWAYS( ( iPrimary.iState & ~ETransactionLock ) == 0, Panic( EDbStreamsPendingOnRollback ) );
261 Database().Revert( LockState() );
262 Database().Abandon( LockState() );
263 if ( LockState() >= EDbWriteLock )
265 Unlock( RDbNotifier::ERollback );
268 // explicit transactions
270 void RDbTransaction::BeginL( const CDbObject& aObject )
272 // begin a user transaction. This first gains a shared read-lock
276 __ASSERT_ALWAYS( GetLock( aObject ) == 0, Panic( EDbBeginNestedTransaction ) );
278 PrepareSLockL( aObject, TUint( ETransactionLock ) );
279 __ASSERT( iAction == EDbReadLock );
280 __ASSERT( iLockState == EDbReadLock );
284 void RDbTransaction::CommitL( const CDbObject& aObject )
286 // Commit a user transaction and release the lock
287 // All updates must be complete for a write-lock
291 __ASSERT_ALWAYS( InTransaction( aObject ), Panic( EDbNoCurrentTransaction ) );
293 if ( iLockCount > 1 )
295 TLock* lock = GetLock( aObject );
297 __ASSERT_ALWAYS( lock->iState == static_cast<TUint>( ETransactionLock ), Panic( EDbStreamsPendingOnCommit ) );
302 __ASSERT_ALWAYS( iAction == EDbReadLock, Panic( EDbUpdatesPendingOnCommit ) );
307 void RDbTransaction::Rollback( const CDbObject& aObject )
309 // Rollback a user transaction and release the lock
310 // All updates must be complete/aborted for a write-lock
314 __ASSERT_ALWAYS( InTransaction( aObject ), Panic( EDbNoCurrentTransaction ) );
315 if ( iLockCount > 1 )
317 TLock* lock = GetLock( aObject );
319 __ASSERT_ALWAYS( lock->iState == static_cast<TUint>( ETransactionLock ), Panic( EDbStreamsPendingOnRollback ) );
324 __ASSERT_ALWAYS( iAction == EDbReadLock, Panic( EDbUpdatesPendingOnRollback ) );
329 void RDbTransaction::PrepareSLockL( const CDbObject& aObject, TUint aInitState )
331 // prepare to acquire a shared read lock
332 // if any holder has an exclusive lock this fails
335 __ASSERT( GetLock( aObject ) == 0 ); // cannot get a 2nd shared lock
337 THolder h = aObject.Context();
338 if ( iLockCount == 0 )
340 iPrimary.iHolder = h; // first lock, no other checks required
341 iPrimary.iState = aInitState;
343 else if ( iLockState != EDbReadLock )
344 __LEAVE( KErrLocked );
346 { // allocate a Sharers-slot
347 TLock* share = iSharers;
348 if ( iLockCount == iMaxLock )
350 TInt newsize = iMaxLock + ELockListGranularity;
351 if ( newsize > EMaxLock )
353 __LEAVE( KErrLocked );
356 iSharers = share = ( TLock* )User::ReAllocL( share, ( newsize - 1 ) * sizeof( TLock ) );
357 iMaxLock = TUint8( newsize );
359 share += iLockCount - 1;
361 share->iState = aInitState;
365 void RDbTransaction::PrepareXLockL( const CDbObject& aObject )
367 // prepare to acquire an exclusive lock
368 // if any other holder has a lock this fails
371 THolder h = aObject.Context();
372 switch ( iLockCount )
374 case 0: // no other holders, acquire the lock
375 iPrimary.iHolder = h;
376 iPrimary.iState = 0; // this is not a transaction lock
378 case 1: // check we are the single Lock holder
379 if (iPrimary.iHolder != h)
380 __LEAVE( KErrLocked );
382 default: // cannot get XLock
383 __LEAVE( KErrLocked );
388 void RDbTransaction::Unlock( RDbNotifier::TEvent aEvent )
390 // Remove the last lock and signal an event to the Notifier
393 __ASSERT( iLockCount == 1 );
394 __ASSERT( ( iPrimary.iState & ~ETransactionLock ) == 0 );
395 TDbLockType ls = LockState();
396 Event( ls == EDbReadLock || ls == EDbXReadLock ? RDbNotifier::EUnlock : aEvent );
398 iAction = iLockState = EDbReadLock;
400 Database().CheckIdle();
403 void RDbTransaction::Unlock( RDbTransaction::TLock& aLock )
405 // Remove a shared lock holder from the list
408 __ASSERT( iLockCount > 1 );
409 __ASSERT( LockState() == EDbReadLock );
410 __ASSERT( ( aLock.iState & ~ETransactionLock ) == 0 );
411 aLock = iSharers[--iLockCount - 1];
414 RDbTransaction::TLock* RDbTransaction::GetLock( const CDbObject& aObject )
416 // Test if aObject holds any lock, and return it
419 const THolder h = aObject.Context();
420 TInt lc = iLockCount;
423 if ( iPrimary.iHolder == h )
427 TLock* const base = iSharers;
428 TLock* l = base + lc;
430 if ( ( --l )->iHolder == h )
432 } while ( l > base );
438 TBool RDbTransaction::InTransaction( const CDbObject& aObject )
440 // Test if aObject holds a non-auto transaction
444 TLock* lock = GetLock( aObject );
445 return lock ? lock->iState & static_cast<TUint>( ETransactionLock ) : 0;
448 void RDbTransaction::ReadPrepareL( const CDbObject& aObject )
450 // Check that aObject can gain a shared read lock and allocate required resources
454 if ( GetLock( aObject ) == 0 )
455 PrepareSLockL( aObject, 0 ); // prepare a S-Lock for the read
456 else if ( iAction == EDbCompactLock ) // Cannot already hold a compaction lock
457 __LEAVE( KErrAccessDenied );
460 void RDbTransaction::ReadBegin( const CDbObject& aObject )
462 // Take a read-lock: ReadPrepareL(aObject) _must_ already have been called
466 TLock* lock = GetLock( aObject );
470 lock = GetLock( aObject );
476 void RDbTransaction::ReadRelease( const CDbObject& aObject )
479 TLock* lock = GetLock( aObject );
481 __ASSERT( ( lock->iState & ~ETransactionLock ) > 0 );
482 if ( --lock->iState == 0 )
483 { // not transaction-lock
484 if ( iLockCount > 1 )
486 else if ( iAction == EDbReadLock ) // no other locks to this client
487 Unlock( RDbNotifier::EUnlock );
491 void RDbTransaction::DMLCheckL()
493 // Check that we can open a new rowset
498 if ( iAction > EDbCompactLock )
499 __LEAVE( KErrAccessDenied );
502 void RDbTransaction::DMLPrepareL( const CDbObject& aObject )
504 // Check that we can do DML, this should be called immediately prior to DMLBegin
508 PrepareXLockL( aObject );
509 if ( iAction>EDbWriteLock )
510 __LEAVE( KErrAccessDenied );
513 void RDbTransaction::DMLBegin()
515 // A Rowset begins an update
519 __ASSERT( iAction == EDbReadLock || iAction == EDbWriteLock );
520 __ASSERT( ( iLockState & EFailed ) == 0 );
521 __ASSERT( iLockCount <= 1 );
522 if ( iAction == EDbReadLock )
523 iAction = EDbWriteLock;
524 if (iLockState == EDbReadLock )
525 iLockState = EDbXReadLock; // escalate lock to exclusive as we are now writing
530 void RDbTransaction::DMLTouch()
532 // This must be called prior to putting DML updates
535 __ASSERT( iAction == EDbWriteLock );
536 __ASSERT( iUpdaters > 0 );
537 TInt ls = iLockState;
538 if ( ls == EDbXReadLock )
539 ls = EDbWriteLock | EFailed;
542 iLockState = TUint8( ls );
545 void RDbTransaction::DMLBeginLC()
548 CleanupStack::PushL( TCleanupItem( DMLAbandon, this ) );
552 void RDbTransaction::DMLCommitL()
554 // A rowset has completed an update
558 __ASSERT( iAction == EDbWriteLock && ( iLockState & EFailed ) );
559 TInt updaters = iUpdaters - 1;
562 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
564 __ASSERT( iLockState == ( EDbWriteLock | EFailed ) );
565 DoCommitL(); // automatic write-commit, release auto-lock
568 iAction = EDbReadLock;
570 iUpdaters = updaters;
571 iLockState &= ~EFailed;
574 void RDbTransaction::DMLRollback()
576 // Rollback a DML operation
580 __ASSERT( iAction == EDbWriteLock );
581 TInt updates = iUpdaters - 1;
584 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
586 __ASSERT( LockState() == EDbWriteLock || LockState() == EDbXReadLock );
587 DoRollback(); // automatic rollback now (may panic)
590 iAction = EDbReadLock;
595 void RDbTransaction::DMLAbandon( TAny* aPtr )
597 STATIC_CAST( RDbTransaction*, aPtr )->DMLRollback();
600 void RDbTransaction::DDLPrepareL( const CDbObject& aObject )
602 // Check that we can use the database for ddl and flush out any tables
603 // should be called before DDLBegin
608 PrepareXLockL( aObject );
609 if ( iAction != EDbReadLock || ( IsLocked() && iPrimary.iState != static_cast<TUint>( ETransactionLock ) ) )
610 __LEAVE( KErrAccessDenied ); // Cannot take sole ownership of the database
611 TInt ls = iLockState;
612 if ( ls >= EDbWriteLock )
613 { // ensure all table data is flushed as they may be "released"
614 iLockState = TUint8( ls | EFailed );
615 Database().FlushL( EDbWriteLock );
616 iLockState = TUint8( ls );
620 void RDbTransaction::DDLBegin()
622 // A DDL object is about to start ops
626 __ASSERT( iAction == EDbReadLock );
627 __ASSERT( ( iLockState & EFailed ) == 0 );
628 __ASSERT( iLockCount <= 1 );
629 iLockState = iAction = EDbSchemaLock;
633 void RDbTransaction::DDLBeginLC()
636 CleanupStack::PushL( TCleanupItem( DDLAbandon, this ) );
639 void RDbTransaction::DDLCommitL()
641 // A DDL incremental object has completed
645 __ASSERT( iAction == EDbSchemaLock );
646 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
648 __ASSERT( iLockState == EDbSchemaLock );
649 DoCommitL(); // release auto-lock
652 iAction = EDbReadLock;
655 void RDbTransaction::DDLRollback()
657 // Rollback a DDL operation
661 __ASSERT( iAction == EDbSchemaLock );
662 iLockState |= EFailed;
663 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
665 __ASSERT( iLockState == ( EDbSchemaLock | EFailed ) );
666 DoRollback(); // release auto-lock
669 iAction = EDbReadLock;
672 void RDbTransaction::DDLAbandon( TAny* aPtr )
674 STATIC_CAST( RDbTransaction*, aPtr )->DDLRollback();
677 // recovery. Nothing else can be done at the same time as this
679 void RDbTransaction::UtilityPrepareL( const CDbObject& aObject )
681 // Check that we are in a state to run a utility
686 PrepareXLockL( aObject );
687 if ( IsLocked() ) // utilities not allowed in user transaction
688 __LEAVE( KErrAccessDenied );
691 void RDbTransaction::UtilityBegin( CDbDatabase::TUtility aType )
693 // Database Recovery object is about to start
697 __ASSERT( !IsLocked() );
698 if ( aType == CDbDatabase::ERecover )
699 iLockState = iAction = EDbRecoveryLock;
701 iLockState = iAction = EDbCompactLock;
705 void RDbTransaction::UtilityCommitL()
707 // Database Recovery has completed
711 Database().SynchL( LockState() );
712 Unlock( iAction == EDbRecoveryLock ? RDbNotifier::ERecover : RDbNotifier::EUnlock ); // release auto-lock
715 void RDbTransaction::UtilityRollback()
718 Database().Revert( LockState() );
719 Unlock( RDbNotifier::EUnlock ); // release auto-lock
722 CDbNotifier* RDbTransaction::NotifierL()
724 // Only support a single notifier for the database (server multiplexes)
728 __LEAVE( KErrNotSupported );
729 return iNotifier = new( ELeave ) CNotifier( *this );
732 void RDbTransaction::Event( RDbNotifier::TEvent aEvent )
734 // Report an event to the Notifier
735 // If the lock was less than a write lock, report unlock only: no commit or rollback
739 // iNotifier->Event( aEvent );