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
70 __ASSERT( iPending >= 0 );
72 if ( iPending > RDbNotifier::EUnlock )
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
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
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 ) );
158 __ASSERT( iUpdaters > 0 );
164 __ASSERT( iLockCount == 1 ); // exactly one lock allowed
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 ) );
174 __ASSERT( iUpdaters > 0 );
175 __ASSERT( ( iLockState & EState ) == EDbWriteLock || ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) );
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 )
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__
226 #define __INVARIANT ( (void)0 )
227 #define __INVARIANT_L ( (void)0 )
229 #endif // _ASSERTIONS
231 inline TDbLockType RDbTransaction::LockState() const
232 { return TDbLockType( iLockState & EState ); }
234 void RDbTransaction::Close()
236 __ASSERT( !IsLocked() );
237 User::Free( iSharers );
238 Event( RDbNotifier::EClose );
241 void RDbTransaction::DoCommitL()
243 // Commit any changes
246 __ASSERT_ALWAYS( ( iPrimary.iState & ~ETransactionLock ) == 0, Panic( EDbStreamsPendingOnCommit ) );
247 iLockState |= EFailed;
248 Database().FlushL( LockState() );
249 Database().SynchL( LockState() );
250 Unlock( RDbNotifier::ECommit );
253 void RDbTransaction::DoRollback()
255 // Rollback any changes
258 __ASSERT_ALWAYS( ( iPrimary.iState & ~ETransactionLock ) == 0, Panic( EDbStreamsPendingOnRollback ) );
259 Database().Revert( LockState() );
260 Database().Abandon( LockState() );
261 if ( LockState() >= EDbWriteLock )
263 Unlock( RDbNotifier::ERollback );
266 // explicit transactions
268 void RDbTransaction::BeginL( const CDbObject& aObject )
270 // begin a user transaction. This first gains a shared read-lock
274 __ASSERT_ALWAYS( GetLock( aObject ) == 0, Panic( EDbBeginNestedTransaction ) );
276 PrepareSLockL( aObject, TUint( ETransactionLock ) );
277 __ASSERT( iAction == EDbReadLock );
278 __ASSERT( iLockState == EDbReadLock );
282 void RDbTransaction::CommitL( const CDbObject& aObject )
284 // Commit a user transaction and release the lock
285 // All updates must be complete for a write-lock
289 __ASSERT_ALWAYS( InTransaction( aObject ), Panic( EDbNoCurrentTransaction ) );
291 if ( iLockCount > 1 )
293 TLock* lock = GetLock( aObject );
295 __ASSERT_ALWAYS( lock->iState == static_cast<TUint>( ETransactionLock ), Panic( EDbStreamsPendingOnCommit ) );
300 __ASSERT_ALWAYS( iAction == EDbReadLock, Panic( EDbUpdatesPendingOnCommit ) );
305 void RDbTransaction::Rollback( const CDbObject& aObject )
307 // Rollback a user transaction and release the lock
308 // All updates must be complete/aborted for a write-lock
312 __ASSERT_ALWAYS( InTransaction( aObject ), Panic( EDbNoCurrentTransaction ) );
313 if ( iLockCount > 1 )
315 TLock* lock = GetLock( aObject );
317 __ASSERT_ALWAYS( lock->iState == static_cast<TUint>( ETransactionLock ), Panic( EDbStreamsPendingOnRollback ) );
322 __ASSERT_ALWAYS( iAction == EDbReadLock, Panic( EDbUpdatesPendingOnRollback ) );
327 void RDbTransaction::PrepareSLockL( const CDbObject& aObject, TUint aInitState )
329 // prepare to acquire a shared read lock
330 // if any holder has an exclusive lock this fails
333 __ASSERT( GetLock( aObject ) == 0 ); // cannot get a 2nd shared lock
335 THolder h = aObject.Context();
336 if ( iLockCount == 0 )
338 iPrimary.iHolder = h; // first lock, no other checks required
339 iPrimary.iState = aInitState;
341 else if ( iLockState != EDbReadLock )
342 __LEAVE( KErrLocked );
344 { // allocate a Sharers-slot
345 TLock* share = iSharers;
346 if ( iLockCount == iMaxLock )
348 TInt newsize = iMaxLock + ELockListGranularity;
349 if ( newsize > EMaxLock )
351 __LEAVE( KErrLocked );
354 iSharers = share = ( TLock* )User::ReAllocL( share, ( newsize - 1 ) * sizeof( TLock ) );
355 iMaxLock = TUint8( newsize );
357 share += iLockCount - 1;
359 share->iState = aInitState;
363 void RDbTransaction::PrepareXLockL( const CDbObject& aObject )
365 // prepare to acquire an exclusive lock
366 // if any other holder has a lock this fails
369 THolder h = aObject.Context();
370 switch ( iLockCount )
372 case 0: // no other holders, acquire the lock
373 iPrimary.iHolder = h;
374 iPrimary.iState = 0; // this is not a transaction lock
376 case 1: // check we are the single Lock holder
377 if (iPrimary.iHolder != h)
378 __LEAVE( KErrLocked );
380 default: // cannot get XLock
381 __LEAVE( KErrLocked );
386 void RDbTransaction::Unlock( RDbNotifier::TEvent aEvent )
388 // Remove the last lock and signal an event to the Notifier
391 __ASSERT( iLockCount == 1 );
392 __ASSERT( ( iPrimary.iState & ~ETransactionLock ) == 0 );
393 TDbLockType ls = LockState();
394 Event( ls == EDbReadLock || ls == EDbXReadLock ? RDbNotifier::EUnlock : aEvent );
396 iAction = iLockState = EDbReadLock;
398 Database().CheckIdle();
401 void RDbTransaction::Unlock( RDbTransaction::TLock& aLock )
403 // Remove a shared lock holder from the list
406 __ASSERT( iLockCount > 1 );
407 __ASSERT( LockState() == EDbReadLock );
408 __ASSERT( ( aLock.iState & ~ETransactionLock ) == 0 );
409 aLock = iSharers[--iLockCount - 1];
412 RDbTransaction::TLock* RDbTransaction::GetLock( const CDbObject& aObject )
414 // Test if aObject holds any lock, and return it
417 const THolder h = aObject.Context();
418 TInt lc = iLockCount;
421 if ( iPrimary.iHolder == h )
425 TLock* const base = iSharers;
426 TLock* l = base + lc;
428 if ( ( --l )->iHolder == h )
430 } while ( l > base );
436 TBool RDbTransaction::InTransaction( const CDbObject& aObject )
438 // Test if aObject holds a non-auto transaction
442 TLock* lock = GetLock( aObject );
443 return lock ? lock->iState & static_cast<TUint>( ETransactionLock ) : 0;
446 void RDbTransaction::ReadPrepareL( const CDbObject& aObject )
448 // Check that aObject can gain a shared read lock and allocate required resources
452 if ( GetLock( aObject ) == 0 )
453 PrepareSLockL( aObject, 0 ); // prepare a S-Lock for the read
454 else if ( iAction == EDbCompactLock ) // Cannot already hold a compaction lock
455 __LEAVE( KErrAccessDenied );
458 void RDbTransaction::ReadBegin( const CDbObject& aObject )
460 // Take a read-lock: ReadPrepareL(aObject) _must_ already have been called
464 TLock* lock = GetLock( aObject );
468 lock = GetLock( aObject );
474 void RDbTransaction::ReadRelease( const CDbObject& aObject )
477 TLock* lock = GetLock( aObject );
479 __ASSERT( ( lock->iState & ~ETransactionLock ) > 0 );
480 if ( --lock->iState == 0 )
481 { // not transaction-lock
482 if ( iLockCount > 1 )
484 else if ( iAction == EDbReadLock ) // no other locks to this client
485 Unlock( RDbNotifier::EUnlock );
489 void RDbTransaction::DMLCheckL()
491 // Check that we can open a new rowset
496 if ( iAction > EDbCompactLock )
497 __LEAVE( KErrAccessDenied );
500 void RDbTransaction::DMLPrepareL( const CDbObject& aObject )
502 // Check that we can do DML, this should be called immediately prior to DMLBegin
506 PrepareXLockL( aObject );
507 if ( iAction>EDbWriteLock )
508 __LEAVE( KErrAccessDenied );
511 void RDbTransaction::DMLBegin()
513 // A Rowset begins an update
517 __ASSERT( iAction == EDbReadLock || iAction == EDbWriteLock );
518 __ASSERT( ( iLockState & EFailed ) == 0 );
519 __ASSERT( iLockCount <= 1 );
520 if ( iAction == EDbReadLock )
521 iAction = EDbWriteLock;
522 if (iLockState == EDbReadLock )
523 iLockState = EDbXReadLock; // escalate lock to exclusive as we are now writing
528 void RDbTransaction::DMLTouch()
530 // This must be called prior to putting DML updates
533 __ASSERT( iAction == EDbWriteLock );
534 __ASSERT( iUpdaters > 0 );
535 TInt ls = iLockState;
536 if ( ls == EDbXReadLock )
537 ls = EDbWriteLock | EFailed;
540 iLockState = TUint8( ls );
543 void RDbTransaction::DMLBeginLC()
546 CleanupStack::PushL( TCleanupItem( DMLAbandon, this ) );
550 void RDbTransaction::DMLCommitL()
552 // A rowset has completed an update
556 __ASSERT( iAction == EDbWriteLock && ( iLockState & EFailed ) );
557 TInt updaters = iUpdaters - 1;
560 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
562 __ASSERT( iLockState == ( EDbWriteLock | EFailed ) );
563 DoCommitL(); // automatic write-commit, release auto-lock
566 iAction = EDbReadLock;
568 iUpdaters = updaters;
569 iLockState &= ~EFailed;
572 void RDbTransaction::DMLRollback()
574 // Rollback a DML operation
578 __ASSERT( iAction == EDbWriteLock );
579 TInt updates = iUpdaters - 1;
582 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
584 __ASSERT( LockState() == EDbWriteLock || LockState() == EDbXReadLock );
585 DoRollback(); // automatic rollback now (may panic)
588 iAction = EDbReadLock;
593 void RDbTransaction::DMLAbandon( TAny* aPtr )
595 STATIC_CAST( RDbTransaction*, aPtr )->DMLRollback();
598 void RDbTransaction::DDLPrepareL( const CDbObject& aObject )
600 // Check that we can use the database for ddl and flush out any tables
601 // should be called before DDLBegin
606 PrepareXLockL( aObject );
607 if ( iAction != EDbReadLock || ( IsLocked() && iPrimary.iState != static_cast<TUint>( ETransactionLock ) ) )
608 __LEAVE( KErrAccessDenied ); // Cannot take sole ownership of the database
609 TInt ls = iLockState;
610 if ( ls >= EDbWriteLock )
611 { // ensure all table data is flushed as they may be "released"
612 iLockState = TUint8( ls | EFailed );
613 Database().FlushL( EDbWriteLock );
614 iLockState = TUint8( ls );
618 void RDbTransaction::DDLBegin()
620 // A DDL object is about to start ops
624 __ASSERT( iAction == EDbReadLock );
625 __ASSERT( ( iLockState & EFailed ) == 0 );
626 __ASSERT( iLockCount <= 1 );
627 iLockState = iAction = EDbSchemaLock;
631 void RDbTransaction::DDLBeginLC()
634 CleanupStack::PushL( TCleanupItem( DDLAbandon, this ) );
637 void RDbTransaction::DDLCommitL()
639 // A DDL incremental object has completed
643 __ASSERT( iAction == EDbSchemaLock );
644 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
646 __ASSERT( iLockState == EDbSchemaLock );
647 DoCommitL(); // release auto-lock
650 iAction = EDbReadLock;
653 void RDbTransaction::DDLRollback()
655 // Rollback a DDL operation
659 __ASSERT( iAction == EDbSchemaLock );
660 iLockState |= EFailed;
661 if ( ( iPrimary.iState & static_cast<TUint>( ETransactionLock ) ) == 0 )
663 __ASSERT( iLockState == ( EDbSchemaLock | EFailed ) );
664 DoRollback(); // release auto-lock
667 iAction = EDbReadLock;
670 void RDbTransaction::DDLAbandon( TAny* aPtr )
672 STATIC_CAST( RDbTransaction*, aPtr )->DDLRollback();
675 // recovery. Nothing else can be done at the same time as this
677 void RDbTransaction::UtilityPrepareL( const CDbObject& aObject )
679 // Check that we are in a state to run a utility
684 PrepareXLockL( aObject );
685 if ( IsLocked() ) // utilities not allowed in user transaction
686 __LEAVE( KErrAccessDenied );
689 void RDbTransaction::UtilityBegin( CDbDatabase::TUtility aType )
691 // Database Recovery object is about to start
695 __ASSERT( !IsLocked() );
696 if ( aType == CDbDatabase::ERecover )
697 iLockState = iAction = EDbRecoveryLock;
699 iLockState = iAction = EDbCompactLock;
703 void RDbTransaction::UtilityCommitL()
705 // Database Recovery has completed
709 Database().SynchL( LockState() );
710 Unlock( iAction == EDbRecoveryLock ? RDbNotifier::ERecover : RDbNotifier::EUnlock ); // release auto-lock
713 void RDbTransaction::UtilityRollback()
716 Database().Revert( LockState() );
717 Unlock( RDbNotifier::EUnlock ); // release auto-lock
720 CDbNotifier* RDbTransaction::NotifierL()
722 // Only support a single notifier for the database (server multiplexes)
726 __LEAVE( KErrNotSupported );
727 return iNotifier = new( ELeave ) CNotifier( *this );
730 void RDbTransaction::Event( RDbNotifier::TEvent aEvent )
732 // Report an event to the Notifier
733 // If the lock was less than a write lock, report unlock only: no commit or rollback
737 iNotifier->Event( aEvent );