Update contrib.
1 // Copyright (c) 2008-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.
24 #include "EComDebug.h"
25 #include "ImplementationInformation.h" // CompareTUidValues
27 #include "resolvercache.h"
28 #include "EComPatchDataConstantv2.h"
30 const TInt KCacheQueueGranularity = 4;
33 // TCounterTicks class
35 inline TCounterTicks::TCounterTicks(TUint aTicks)
40 /** This substraction calculate elapsed ticks: aEndTick - this */
42 TUint TCounterTicks::ElapsedSinceThis(TUint aEndTick) const
45 if (aEndTick < iTicks) // wrap around occurred
47 diff = KMaxTUint - iTicks + aEndTick + 1;
51 diff = aEndTick - iTicks;
56 /** default constructor which initializes all member data to 0. */
57 RResolverCacheEntry::RResolverCacheEntry()
58 : iLastUse( TCounterTicks(0) ), iLruRank(0), iResolverUid(KNullUid),
59 iNewLFuncPtr(NULL), iFlags(EEntryFlagsNone)
61 iLibrary.SetHandle(KNullHandle);
64 /** construct RResolverCacheEntry with specific data
65 @param aResolverUid TUid of the resolver.
66 @param aLib RLibrary with handle open on the resolver DLL.
67 @param aNewL Function ptr to instantiate the resolver.
68 @param aFlags Special conditions about the cache entry.
70 RResolverCacheEntry::RResolverCacheEntry(const TUid aResolverUid,
74 : iLastUse( TCounterTicks(0) ), iLruRank(0), iResolverUid(aResolverUid),
75 iNewLFuncPtr(aNewL), iFlags(aFlags)
77 iLibrary.SetHandle(aLib.Handle());
80 /** unallocate resources owned by the RResolverCacheEntry object */
81 void RResolverCacheEntry::Close()
86 /** This method compares two RResolverCacheEntry objects.
87 @param aEntry1 first RResolverCacheEntry object.
88 @param aEntry2 second RResolverCacheEntry object.
89 @return 0 means the two objects are equal.
90 positive value means first object is greater than the second.
91 negative vlaue means second object is greater than the first.
93 TInt RResolverCacheEntry::CompareUid(const RResolverCacheEntry& aEntry1,
94 const RResolverCacheEntry& aEntry2)
96 return CompareTUidValues(aEntry1.iResolverUid.iUid,
97 aEntry2.iResolverUid.iUid);
100 /** Compare the age of two cache entries.
101 @param aOther The RResolverCacheEntry to compare with.
102 @param aCurrTick The current system tick. It simplifies handling
103 of system tick wrap around to zero.
104 @return ETrue means "this" object is older than aOther.
105 EFalse means aOther is older than "this".
107 TBool RResolverCacheEntry::ThisIsOlder(const RResolverCacheEntry& aOther,
108 TUint aCurrTick) const
110 if (iLastUse.iTicks == aOther.iLastUse.iTicks)
112 return iLruRank < aOther.iLruRank;
115 // Because of counter wrap around, it is not safe to directly
116 // compare the two ticks. Drag in the current system tick.
117 return iLastUse.ElapsedSinceThis(aCurrTick) >
118 aOther.iLastUse.ElapsedSinceThis(aCurrTick);
121 //=================================
122 // CCustomResolverCache class
123 //=================================
125 /** CCustomResolverCache constructor
126 @param aCacheSize Maximum number of entries allowed to cache.
128 CCustomResolverCache::CCustomResolverCache(TUint32 aCacheSize)
129 : CTimer(CActive::EPriorityStandard),
130 iResolvers(KCacheQueueGranularity),
131 iMaxCacheSize(aCacheSize)
135 /** static factory method to instantiate CCustomResolverCache
136 @param aCacheSize Maximum number of entries allowed to cache.
137 @param aCacheTimeout Cache timeout in microseconds.
138 @leave Any of the system wide error codes.
140 CCustomResolverCache* CCustomResolverCache::NewL(TUint32 aCacheSize, TUint32 aCacheTimeout)
142 CCustomResolverCache* self = new (ELeave) CCustomResolverCache(aCacheSize);
143 CleanupStack::PushL(self);
144 self->ConstructL(aCacheTimeout);
145 CleanupStack::Pop(self);
149 /** Standard two phase construction to complete construction of
150 the CCustomResolverCache object.
151 @param aCacheTimeout Cache timeout in microseconds.
152 @leave Any of the system wide error codes.
154 void CCustomResolverCache::ConstructL(TUint32 aCacheTimeout)
156 User::LeaveIfError(HAL::Get(HALData::ESystemTickPeriod,
158 iEntryTimeToLive = (aCacheTimeout + iSystemTickPeriod - 1) / iSystemTickPeriod;
160 CTimer::ConstructL();
161 CActiveScheduler::Add(this);
164 /** CCustomResolverCache destructor */
165 CCustomResolverCache::~CCustomResolverCache()
168 for (TInt i = iResolvers.Count() - 1; i >= 0; i--)
170 iResolvers[i].Close();
175 /** Implement the CActive RunL pure virtual */
176 void CCustomResolverCache::RunL()
178 TUint currTickCount = User::TickCount();
181 for (TInt i = iResolvers.Count() - 1; i >= 0; i--)
183 RResolverCacheEntry& resolverEntry = iResolvers[i];
184 TUint age = resolverEntry.iLastUse.ElapsedSinceThis(currTickCount);
185 if (age >= iEntryTimeToLive)
187 resolverEntry.Close();
188 iResolvers.Remove(i);
190 else if (age > lruAge)
196 if (iResolvers.Count() > 0)
198 After( iSystemTickPeriod * (iEntryTimeToLive - lruAge + 1) );
201 #ifdef __ECOMSERVER_TESTING__
202 // In unit testing notify test bed that cache timer has fired.
203 iTimerExpireCB.CallBack(0, NULL);
207 /** Search of a resolver UID in cache
208 @param aResolverUid the resolver to search for.
209 @return If find is successful, index of the resolver in RArray of cached resolvers.
210 Return KErrNotFound if resolver is not in cache.
212 TInt CCustomResolverCache::FindResolver(const TUid aResolverUid) const
214 RResolverCacheEntry trgt;
215 trgt.iResolverUid = aResolverUid;
216 TLinearOrder<RResolverCacheEntry> comparator(RResolverCacheEntry::CompareUid);
217 return iResolvers.FindInOrder(trgt, comparator);
220 /** Add a resolver library to cache
221 @param aResolverUid Implementation UID of the resolver.
222 @param aLib The RLibrary object which has the resolver loaded. The handle
223 of the RLibrary is owned by the cache if call is successful.
224 @param aNewL value for the iNewLFuncPtr member data of RResolverCacheEntry.
225 @param aFlags value for the iFlags member data of RResolverCacheEntry.
226 @return KErrNone if the data is added to cache.
227 KErrNoMemory if fail to insert the data in RArray.
229 TInt CCustomResolverCache::CacheResolver(const TUid aResolverUid,
234 if (iResolvers.Count() == iMaxCacheSize)
236 EvictLeastRecentlyUsed();
239 RResolverCacheEntry entry(aResolverUid, aLib, aNewL, aFlags);
240 SetLastUseTime(entry);
243 __ECOM_TRACE2("ECOM: adding custom resolver 0x%X to cache. New queue size will be %d.\n", aResolverUid.iUid, 1 + iResolvers.Count());
246 TLinearOrder<RResolverCacheEntry> comparator(RResolverCacheEntry::CompareUid);
247 TInt err = iResolvers.InsertInOrder(entry, comparator);
249 // if cache was empty before need to start timer
250 if (err == KErrNone && ! IsActive())
252 After( iSystemTickPeriod * (iEntryTimeToLive + 1) );
257 /** Check if both queue size and cache timeout are non zero */
258 TBool CCustomResolverCache::CachingEnabled() const
260 return (iMaxCacheSize && iEntryTimeToLive);
263 /** Remove a cached entry
264 @param aIndex position of entry in the array.
266 void CCustomResolverCache::Remove(TInt aIndex)
268 iResolvers[aIndex].Close();
269 iResolvers.Remove(aIndex);
272 /** Search for a resolverUID. If found, return the NewL pointer and
273 update time to live of the entry.
274 @param aResolverUid the resolver to lookup
275 @param aNewLFuncPtr output parameter. If lookup successful it has
276 the function pointer to instantiate the resolver.
277 @return True if resolver is in cache. False otherwise.
278 @post If cache hit, the timestamp of the entry is updated.
280 TBool CCustomResolverCache::CacheLookup(const TUid aResolverUid,
281 TProxyNewLPtr& aNewLFuncPtr)
283 TInt i = FindResolver(aResolverUid);
286 aNewLFuncPtr = iResolvers[i].iNewLFuncPtr;
287 SetLastUseTime(iResolvers[i]);
292 /** Remove a resolver from cache.
293 @param aResolverUid Identify the resolver to remove.
294 @return ETrue if aResolverUid is found in cache. EFalse means the
295 resolver is not in cache.
297 TBool CCustomResolverCache::Remove(const TUid aResolverUid)
299 TInt i = FindResolver(aResolverUid);
307 /** Remove cached entries with flags set.
308 @param aMask If an entry has any of the bits in aMask set, it is removed.
310 void CCustomResolverCache::RemoveItemsWithFlags(TUint32 aMask)
312 for (TInt i = iResolvers.Count() - 1; i >= 0; i--)
314 if (iResolvers[i].iFlags & aMask)
321 /** evict the least recently used entry in cache
323 void CCustomResolverCache::EvictLeastRecentlyUsed()
325 TUint curr = User::TickCount();
326 TInt index = 0; // set to first entry in cache.
328 for (TInt i = 1; i < iResolvers.Count(); i++)
330 RResolverCacheEntry& resolverEntry = iResolvers[i];
331 if (resolverEntry.ThisIsOlder(iResolvers[index], curr))
340 /** Set the iLastUse field and iLruRank field of the entry.
341 The iLruRank field serves as tie breaker when two entries are
342 added within the same tick period.
343 @param aEntry The entry to update.
345 void CCustomResolverCache::SetLastUseTime(RResolverCacheEntry& aEntry)
347 TUint curr = User::TickCount();
348 aEntry.iLastUse.iTicks = curr;
351 if (curr == iMostRecentTimestamp)
353 // There are entries with same timestamp. So have to step
354 // through each cache entry to resolve senority.
355 for (TInt i = 0; i < iResolvers.Count(); i++)
357 RResolverCacheEntry& another = iResolvers[i];
358 if (another.iLastUse.iTicks == curr &&
359 another.iResolverUid != aEntry.iResolverUid &&
360 another.iLruRank >= aEntry.iLruRank)
362 aEntry.iLruRank = another.iLruRank + 1;
368 iMostRecentTimestamp = curr;