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.
22 NONSHARABLE_CLASS(RDbCache::CCache) : private CBase
25 static CCache* OpenL( TInt aSize, TBool aUseTimer );
29 void Hold( CBase* aObject, TUint aMicroSeconds );
30 void Release ( const CBase& aObject );
38 enum { ETimerPriority = -10 };
39 enum { ETimerPeriod = 0x100000 }; // ~1.0s
41 static inline TInt TlsHandle();
45 static CCache* NewL( TInt aSize, TBool aUseTimer );
47 void Expire( TInt aElapsedTime );
48 void Remove( TEntry*& aRef );
50 static void DoFlush( TAny* aPtr );
54 TTimeIntervalMicroSeconds32 iTickPeriod;
58 TEntry iEntries[1]; // or maybe more
61 // Class CDbObjectCache
63 inline TInt RDbCache::CCache::TlsHandle()
64 // use the address of a static function for the handle
65 { return TInt( NewL ); }
68 RDbCache::CCache::CCache( TInt aSize )
70 // Initialise the free entry list
73 TEntry* entry = iEntries;
74 while ( --aSize != 0 )
76 entry[1].iNext = entry;
82 RDbCache::CCache::~CCache()
85 // empty the cache (destory the items now)
87 __ASSERT( iCache == 0 );
89 UserSvr::DllFreeTls( TlsHandle() );
92 RDbCache::CCache* RDbCache::CCache::NewL( TInt aSize, TBool aUseTimer )
94 // Construct a cache with aSize slots and one referee
97 CCache* cache = new( ELeave, sizeof( TEntry ) * ( aSize - 1 ) ) CCache( aSize ); // get the extra size for the cache entries, leaves on error
98 CleanupClosePushL( *cache );
99 User::LeaveIfError( UserHal::TickPeriod( cache->iTickPeriod ) );
100 User::LeaveIfError( UserSvr::DllSetTls( TlsHandle(), cache ) );
102 cache->iTimer = CPeriodic::NewL( ETimerPriority );
107 inline void RDbCache::CCache::Open()
111 void RDbCache::CCache::Close()
113 // remove a referee and delete as required
116 __ASSERT( iRef >= 0 );
121 RDbCache::CCache* RDbCache::CCache::OpenL( TInt aSize, TBool aUseTimer )
123 // Grab a reference to the cache, constructing it if required
126 CCache* cache = ( CCache* )UserSvr::DllTls( TlsHandle() );
128 return NewL( aSize, aUseTimer );
133 void RDbCache::CCache::Hold( CBase* aObject, TUint aMicroSeconds )
135 // Hold aObject in the cache or destroy it
138 Flush(); // Destroy expired entries and re-assess Zero-time
139 TInt ticks = aMicroSeconds / TUint( iTickPeriod.Int() );
140 TEntry* entry = iFree;
142 { // no free slots: check the first cache entry
144 if ( iCache->iDelta > ticks )
145 { // aObject expires first
149 ExpireFirst(); // remove the first entry and use it
152 iFree = entry->iNext; // move the free list pointer to the next entry
154 // find the insertion point in the cache delta-list
155 TEntry** pcache = &iCache;
159 __ASSERT( ticks >= 0 );
163 TInt t = ticks - cache->iDelta;
165 { // add to the list here
166 cache->iDelta = -t; // reduce the following delta
169 ticks = t; // reduce the entry delta
170 pcache = &cache->iNext;
172 *pcache = entry; // set up the entry
173 entry->iDelta = ticks;
174 entry->iNext = cache;
175 entry->iObject = aObject;
176 // kick the timer if we need to
177 if ( iTimer && !iTimer->IsActive() )
178 iTimer->Start( ETimerPeriod, ETimerPeriod, TCallBack( ( TInt (*)(TAny*) )DoFlush, this ) );
181 void RDbCache::CCache::Remove( RDbCache::CCache::TEntry*& aRef )
183 // Remove the entry at aRef from the cache
186 TEntry& entry = *aRef;
187 TEntry* next = entry.iNext;
192 next->iDelta += entry.iDelta;
193 else if ( iTimer ) // the cache is now empty, so stop the timer if we have one
197 void RDbCache::CCache::ExpireFirst()
199 // Expire the first entry in the cache
202 __ASSERT( iCache != 0 );
203 // the ordering here is important. Removing the entry first allows the
204 // object d'tor to call Release() without causing re-entrancy problems.
205 CBase* object = iCache->iObject;
210 void RDbCache::CCache::Release( const CBase& aObject )
212 // Remove the cache entry for aObject, if it is in the cache
215 TEntry** pcache = &iCache;
218 TEntry* entry = *pcache;
221 if ( entry->iObject == &aObject )
226 pcache = &entry->iNext;
230 void RDbCache::CCache::Expire( TInt aElapsedTime )
232 // Destroy entries which expire with aElapsedTime
235 __ASSERT( aElapsedTime > 0 );
236 if ( iCache && ( iCache->iDelta -= aElapsedTime ) < 0 )
238 Open(); // This allows the cache to be owned by an object in the cache
240 while ( iCache && iCache->iDelta < 0 );
241 Close(); // The cache may be destroyed now
245 void RDbCache::CCache::Flush()
247 // Check the execution clock and destroy any expired entries
249 // Care has to be taken to handle the 32-bit wraparound of the tick-count
250 // e.g. iZeroTime = 0xffffffffu, now = 0
253 TUint now = User::TickCount();
254 TUint elapsed = now - iZeroTime;
257 Expire( elapsed <= TUint( KMaxTInt ) ? elapsed : TUint( KMaxTInt ) );
260 void RDbCache::CCache::DoFlush( TAny* aPtr )
262 // Callback for the timer
265 static_cast<CCache*>( aPtr )->Flush();
271 TInt RDbCache::Open( TInt aSize, TBool aUseTimer )
273 // Get a handle on the cache
276 __ASSERT( aSize > 0 );
277 TRAPD( r, iCache = CCache::OpenL( aSize, aUseTimer ) );
281 void RDbCache::Close()
283 // Close this handle on the cache
286 CCache* cache = iCache;
294 void RDbCache::Hold( CBase* aObject, TUint aMicroSeconds )
296 // Hold aObject on the cache, if open
297 // We are now responsible for deleting the object
301 iCache->Hold( aObject, aMicroSeconds );
303 delete aObject; // no cache available
306 void RDbCache::Release( const CBase& aObject ) const
308 // Retrieve aObject from the cache
312 iCache->Release( aObject );
315 void RDbCache::Flush()
317 // Destroy any cached objects which have expired