sl@0
|
1 |
// Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
// DBMS object cache
|
sl@0
|
15 |
//
|
sl@0
|
16 |
//
|
sl@0
|
17 |
|
sl@0
|
18 |
#include "UD_STD.H"
|
sl@0
|
19 |
#include "D32CACHE.H"
|
sl@0
|
20 |
#include <e32svr.h>
|
sl@0
|
21 |
|
sl@0
|
22 |
NONSHARABLE_CLASS(RDbCache::CCache) : private CBase
|
sl@0
|
23 |
{
|
sl@0
|
24 |
public:
|
sl@0
|
25 |
static CCache* OpenL( TInt aSize, TBool aUseTimer );
|
sl@0
|
26 |
void Close();
|
sl@0
|
27 |
//
|
sl@0
|
28 |
void Flush();
|
sl@0
|
29 |
void Hold( CBase* aObject, TUint aMicroSeconds );
|
sl@0
|
30 |
void Release ( const CBase& aObject );
|
sl@0
|
31 |
private:
|
sl@0
|
32 |
struct TEntry
|
sl@0
|
33 |
{
|
sl@0
|
34 |
TEntry* iNext;
|
sl@0
|
35 |
TInt iDelta;
|
sl@0
|
36 |
CBase* iObject;
|
sl@0
|
37 |
};
|
sl@0
|
38 |
enum { ETimerPriority = -10 };
|
sl@0
|
39 |
enum { ETimerPeriod = 0x100000 }; // ~1.0s
|
sl@0
|
40 |
private:
|
sl@0
|
41 |
static inline TInt TlsHandle();
|
sl@0
|
42 |
CCache( TInt aSize );
|
sl@0
|
43 |
~CCache();
|
sl@0
|
44 |
//
|
sl@0
|
45 |
static CCache* NewL( TInt aSize, TBool aUseTimer );
|
sl@0
|
46 |
inline void Open();
|
sl@0
|
47 |
void Expire( TInt aElapsedTime );
|
sl@0
|
48 |
void Remove( TEntry*& aRef );
|
sl@0
|
49 |
void ExpireFirst();
|
sl@0
|
50 |
static void DoFlush( TAny* aPtr );
|
sl@0
|
51 |
private:
|
sl@0
|
52 |
TInt iRef;
|
sl@0
|
53 |
CPeriodic* iTimer;
|
sl@0
|
54 |
TTimeIntervalMicroSeconds32 iTickPeriod;
|
sl@0
|
55 |
TUint iZeroTime;
|
sl@0
|
56 |
TEntry* iCache;
|
sl@0
|
57 |
TEntry* iFree;
|
sl@0
|
58 |
TEntry iEntries[1]; // or maybe more
|
sl@0
|
59 |
};
|
sl@0
|
60 |
|
sl@0
|
61 |
// Class CDbObjectCache
|
sl@0
|
62 |
|
sl@0
|
63 |
inline TInt RDbCache::CCache::TlsHandle()
|
sl@0
|
64 |
// use the address of a static function for the handle
|
sl@0
|
65 |
{ return TInt( NewL ); }
|
sl@0
|
66 |
|
sl@0
|
67 |
|
sl@0
|
68 |
RDbCache::CCache::CCache( TInt aSize )
|
sl@0
|
69 |
//
|
sl@0
|
70 |
// Initialise the free entry list
|
sl@0
|
71 |
//
|
sl@0
|
72 |
{
|
sl@0
|
73 |
TEntry* entry = iEntries;
|
sl@0
|
74 |
while ( --aSize != 0 )
|
sl@0
|
75 |
{
|
sl@0
|
76 |
entry[1].iNext = entry;
|
sl@0
|
77 |
++entry;
|
sl@0
|
78 |
}
|
sl@0
|
79 |
iFree = entry;
|
sl@0
|
80 |
}
|
sl@0
|
81 |
|
sl@0
|
82 |
RDbCache::CCache::~CCache()
|
sl@0
|
83 |
{
|
sl@0
|
84 |
__ASSERT( iRef < 0 );
|
sl@0
|
85 |
// empty the cache (destory the items now)
|
sl@0
|
86 |
Expire( KMaxTInt );
|
sl@0
|
87 |
__ASSERT( iCache == 0 );
|
sl@0
|
88 |
delete iTimer;
|
sl@0
|
89 |
UserSvr::DllFreeTls( TlsHandle() );
|
sl@0
|
90 |
}
|
sl@0
|
91 |
|
sl@0
|
92 |
RDbCache::CCache* RDbCache::CCache::NewL( TInt aSize, TBool aUseTimer )
|
sl@0
|
93 |
//
|
sl@0
|
94 |
// Construct a cache with aSize slots and one referee
|
sl@0
|
95 |
//
|
sl@0
|
96 |
{
|
sl@0
|
97 |
CCache* cache = new( ELeave, sizeof( TEntry ) * ( aSize - 1 ) ) CCache( aSize ); // get the extra size for the cache entries, leaves on error
|
sl@0
|
98 |
CleanupClosePushL( *cache );
|
sl@0
|
99 |
User::LeaveIfError( UserHal::TickPeriod( cache->iTickPeriod ) );
|
sl@0
|
100 |
User::LeaveIfError( UserSvr::DllSetTls( TlsHandle(), cache ) );
|
sl@0
|
101 |
if (aUseTimer)
|
sl@0
|
102 |
cache->iTimer = CPeriodic::NewL( ETimerPriority );
|
sl@0
|
103 |
CleanupStack::Pop();
|
sl@0
|
104 |
return cache;
|
sl@0
|
105 |
}
|
sl@0
|
106 |
|
sl@0
|
107 |
inline void RDbCache::CCache::Open()
|
sl@0
|
108 |
// add a referee
|
sl@0
|
109 |
{ ++iRef; }
|
sl@0
|
110 |
|
sl@0
|
111 |
void RDbCache::CCache::Close()
|
sl@0
|
112 |
//
|
sl@0
|
113 |
// remove a referee and delete as required
|
sl@0
|
114 |
//
|
sl@0
|
115 |
{
|
sl@0
|
116 |
__ASSERT( iRef >= 0 );
|
sl@0
|
117 |
if ( --iRef < 0 )
|
sl@0
|
118 |
delete this;
|
sl@0
|
119 |
}
|
sl@0
|
120 |
|
sl@0
|
121 |
RDbCache::CCache* RDbCache::CCache::OpenL( TInt aSize, TBool aUseTimer )
|
sl@0
|
122 |
//
|
sl@0
|
123 |
// Grab a reference to the cache, constructing it if required
|
sl@0
|
124 |
//
|
sl@0
|
125 |
{
|
sl@0
|
126 |
CCache* cache = ( CCache* )UserSvr::DllTls( TlsHandle() );
|
sl@0
|
127 |
if (!cache)
|
sl@0
|
128 |
return NewL( aSize, aUseTimer );
|
sl@0
|
129 |
cache->Open();
|
sl@0
|
130 |
return cache;
|
sl@0
|
131 |
}
|
sl@0
|
132 |
|
sl@0
|
133 |
void RDbCache::CCache::Hold( CBase* aObject, TUint aMicroSeconds )
|
sl@0
|
134 |
//
|
sl@0
|
135 |
// Hold aObject in the cache or destroy it
|
sl@0
|
136 |
//
|
sl@0
|
137 |
{
|
sl@0
|
138 |
Flush(); // Destroy expired entries and re-assess Zero-time
|
sl@0
|
139 |
TInt ticks = aMicroSeconds / TUint( iTickPeriod.Int() );
|
sl@0
|
140 |
TEntry* entry = iFree;
|
sl@0
|
141 |
if ( entry == 0 )
|
sl@0
|
142 |
{ // no free slots: check the first cache entry
|
sl@0
|
143 |
__ASSERT( iCache );
|
sl@0
|
144 |
if ( iCache->iDelta > ticks )
|
sl@0
|
145 |
{ // aObject expires first
|
sl@0
|
146 |
delete aObject;
|
sl@0
|
147 |
return;
|
sl@0
|
148 |
}
|
sl@0
|
149 |
ExpireFirst(); // remove the first entry and use it
|
sl@0
|
150 |
entry = iFree;
|
sl@0
|
151 |
}
|
sl@0
|
152 |
iFree = entry->iNext; // move the free list pointer to the next entry
|
sl@0
|
153 |
//
|
sl@0
|
154 |
// find the insertion point in the cache delta-list
|
sl@0
|
155 |
TEntry** pcache = &iCache;
|
sl@0
|
156 |
TEntry* cache;
|
sl@0
|
157 |
for ( ; ; )
|
sl@0
|
158 |
{
|
sl@0
|
159 |
__ASSERT( ticks >= 0 );
|
sl@0
|
160 |
cache = *pcache;
|
sl@0
|
161 |
if ( !cache )
|
sl@0
|
162 |
break; // add to end
|
sl@0
|
163 |
TInt t = ticks - cache->iDelta;
|
sl@0
|
164 |
if ( t < 0 )
|
sl@0
|
165 |
{ // add to the list here
|
sl@0
|
166 |
cache->iDelta = -t; // reduce the following delta
|
sl@0
|
167 |
break;
|
sl@0
|
168 |
}
|
sl@0
|
169 |
ticks = t; // reduce the entry delta
|
sl@0
|
170 |
pcache = &cache->iNext;
|
sl@0
|
171 |
}
|
sl@0
|
172 |
*pcache = entry; // set up the entry
|
sl@0
|
173 |
entry->iDelta = ticks;
|
sl@0
|
174 |
entry->iNext = cache;
|
sl@0
|
175 |
entry->iObject = aObject;
|
sl@0
|
176 |
// kick the timer if we need to
|
sl@0
|
177 |
if ( iTimer && !iTimer->IsActive() )
|
sl@0
|
178 |
iTimer->Start( ETimerPeriod, ETimerPeriod, TCallBack( ( TInt (*)(TAny*) )DoFlush, this ) );
|
sl@0
|
179 |
}
|
sl@0
|
180 |
|
sl@0
|
181 |
void RDbCache::CCache::Remove( RDbCache::CCache::TEntry*& aRef )
|
sl@0
|
182 |
//
|
sl@0
|
183 |
// Remove the entry at aRef from the cache
|
sl@0
|
184 |
//
|
sl@0
|
185 |
{
|
sl@0
|
186 |
TEntry& entry = *aRef;
|
sl@0
|
187 |
TEntry* next = entry.iNext;
|
sl@0
|
188 |
entry.iNext = iFree;
|
sl@0
|
189 |
iFree = &entry;
|
sl@0
|
190 |
aRef = next;
|
sl@0
|
191 |
if ( next )
|
sl@0
|
192 |
next->iDelta += entry.iDelta;
|
sl@0
|
193 |
else if ( iTimer ) // the cache is now empty, so stop the timer if we have one
|
sl@0
|
194 |
iTimer->Cancel();
|
sl@0
|
195 |
}
|
sl@0
|
196 |
|
sl@0
|
197 |
void RDbCache::CCache::ExpireFirst()
|
sl@0
|
198 |
//
|
sl@0
|
199 |
// Expire the first entry in the cache
|
sl@0
|
200 |
//
|
sl@0
|
201 |
{
|
sl@0
|
202 |
__ASSERT( iCache != 0 );
|
sl@0
|
203 |
// the ordering here is important. Removing the entry first allows the
|
sl@0
|
204 |
// object d'tor to call Release() without causing re-entrancy problems.
|
sl@0
|
205 |
CBase* object = iCache->iObject;
|
sl@0
|
206 |
Remove( iCache );
|
sl@0
|
207 |
delete object;
|
sl@0
|
208 |
}
|
sl@0
|
209 |
|
sl@0
|
210 |
void RDbCache::CCache::Release( const CBase& aObject )
|
sl@0
|
211 |
//
|
sl@0
|
212 |
// Remove the cache entry for aObject, if it is in the cache
|
sl@0
|
213 |
//
|
sl@0
|
214 |
{
|
sl@0
|
215 |
TEntry** pcache = &iCache;
|
sl@0
|
216 |
for ( ; ; )
|
sl@0
|
217 |
{
|
sl@0
|
218 |
TEntry* entry = *pcache;
|
sl@0
|
219 |
if ( !entry )
|
sl@0
|
220 |
return;
|
sl@0
|
221 |
if ( entry->iObject == &aObject )
|
sl@0
|
222 |
{
|
sl@0
|
223 |
Remove( *pcache );
|
sl@0
|
224 |
return;
|
sl@0
|
225 |
}
|
sl@0
|
226 |
pcache = &entry->iNext;
|
sl@0
|
227 |
}
|
sl@0
|
228 |
}
|
sl@0
|
229 |
|
sl@0
|
230 |
void RDbCache::CCache::Expire( TInt aElapsedTime )
|
sl@0
|
231 |
//
|
sl@0
|
232 |
// Destroy entries which expire with aElapsedTime
|
sl@0
|
233 |
//
|
sl@0
|
234 |
{
|
sl@0
|
235 |
__ASSERT( aElapsedTime > 0 );
|
sl@0
|
236 |
if ( iCache && ( iCache->iDelta -= aElapsedTime ) < 0 )
|
sl@0
|
237 |
{
|
sl@0
|
238 |
Open(); // This allows the cache to be owned by an object in the cache
|
sl@0
|
239 |
do ExpireFirst();
|
sl@0
|
240 |
while ( iCache && iCache->iDelta < 0 );
|
sl@0
|
241 |
Close(); // The cache may be destroyed now
|
sl@0
|
242 |
}
|
sl@0
|
243 |
}
|
sl@0
|
244 |
|
sl@0
|
245 |
void RDbCache::CCache::Flush()
|
sl@0
|
246 |
//
|
sl@0
|
247 |
// Check the execution clock and destroy any expired entries
|
sl@0
|
248 |
//
|
sl@0
|
249 |
// Care has to be taken to handle the 32-bit wraparound of the tick-count
|
sl@0
|
250 |
// e.g. iZeroTime = 0xffffffffu, now = 0
|
sl@0
|
251 |
//
|
sl@0
|
252 |
{
|
sl@0
|
253 |
TUint now = User::TickCount();
|
sl@0
|
254 |
TUint elapsed = now - iZeroTime;
|
sl@0
|
255 |
iZeroTime = now;
|
sl@0
|
256 |
if ( elapsed )
|
sl@0
|
257 |
Expire( elapsed <= TUint( KMaxTInt ) ? elapsed : TUint( KMaxTInt ) );
|
sl@0
|
258 |
}
|
sl@0
|
259 |
|
sl@0
|
260 |
void RDbCache::CCache::DoFlush( TAny* aPtr )
|
sl@0
|
261 |
//
|
sl@0
|
262 |
// Callback for the timer
|
sl@0
|
263 |
//
|
sl@0
|
264 |
{
|
sl@0
|
265 |
static_cast<CCache*>( aPtr )->Flush();
|
sl@0
|
266 |
}
|
sl@0
|
267 |
|
sl@0
|
268 |
|
sl@0
|
269 |
// Class RDbCache
|
sl@0
|
270 |
|
sl@0
|
271 |
TInt RDbCache::Open( TInt aSize, TBool aUseTimer )
|
sl@0
|
272 |
//
|
sl@0
|
273 |
// Get a handle on the cache
|
sl@0
|
274 |
//
|
sl@0
|
275 |
{
|
sl@0
|
276 |
__ASSERT( aSize > 0 );
|
sl@0
|
277 |
TRAPD( r, iCache = CCache::OpenL( aSize, aUseTimer ) );
|
sl@0
|
278 |
return r;
|
sl@0
|
279 |
}
|
sl@0
|
280 |
|
sl@0
|
281 |
void RDbCache::Close()
|
sl@0
|
282 |
//
|
sl@0
|
283 |
// Close this handle on the cache
|
sl@0
|
284 |
//
|
sl@0
|
285 |
{
|
sl@0
|
286 |
CCache* cache = iCache;
|
sl@0
|
287 |
if ( cache )
|
sl@0
|
288 |
{
|
sl@0
|
289 |
iCache = 0;
|
sl@0
|
290 |
cache->Close();
|
sl@0
|
291 |
}
|
sl@0
|
292 |
}
|
sl@0
|
293 |
|
sl@0
|
294 |
void RDbCache::Hold( CBase* aObject, TUint aMicroSeconds )
|
sl@0
|
295 |
//
|
sl@0
|
296 |
// Hold aObject on the cache, if open
|
sl@0
|
297 |
// We are now responsible for deleting the object
|
sl@0
|
298 |
//
|
sl@0
|
299 |
{
|
sl@0
|
300 |
if ( iCache )
|
sl@0
|
301 |
iCache->Hold( aObject, aMicroSeconds );
|
sl@0
|
302 |
else
|
sl@0
|
303 |
delete aObject; // no cache available
|
sl@0
|
304 |
}
|
sl@0
|
305 |
|
sl@0
|
306 |
void RDbCache::Release( const CBase& aObject ) const
|
sl@0
|
307 |
//
|
sl@0
|
308 |
// Retrieve aObject from the cache
|
sl@0
|
309 |
//
|
sl@0
|
310 |
{
|
sl@0
|
311 |
if ( iCache )
|
sl@0
|
312 |
iCache->Release( aObject );
|
sl@0
|
313 |
}
|
sl@0
|
314 |
|
sl@0
|
315 |
void RDbCache::Flush()
|
sl@0
|
316 |
//
|
sl@0
|
317 |
// Destroy any cached objects which have expired
|
sl@0
|
318 |
//
|
sl@0
|
319 |
{
|
sl@0
|
320 |
if ( iCache )
|
sl@0
|
321 |
iCache->Flush();
|
sl@0
|
322 |
}
|
sl@0
|
323 |
|
sl@0
|
324 |
|
sl@0
|
325 |
|