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.
21 TUint8 iData[KPoolPageSize];
24 EXPORT_C CPageCache* CPageCache::NewL(TInt aPages)
25 /** Allocates and constructs a new page cache.
27 @param aPages Number of pages in the cache
28 @return New page cache object */
30 CPageCache* cache=CPageCache::NewLC(aPages);
35 EXPORT_C CPageCache* CPageCache::NewLC(TInt aPages)
36 /** Allocates and constructs a new page cache, and leaves it on the cleanup stack.
38 @param aPages Number of pages in the cache
39 @return New page cache object */
41 CPageCache* cache=new(ELeave) CPageCache;
42 CleanupStack::PushL(cache);
43 cache->ConstructL(aPages);
47 EXPORT_C CPageCache::CPageCache()
48 : iFree(_FOFF(TCachePage,iLink)-_FOFF(SCachePage,iPage[1]))/*,iPages(NULL),iEnd(NULL)*/
49 /** Default constructor. */
51 #if defined(__PAGE_CACHE_STATS)
56 EXPORT_C void CPageCache::ConstructL(TInt aPages)
57 /** Second phase construction.
59 @param aPages Number of pages in the cache */
61 SCachePage* page=new(ELeave) SCachePage[aPages];
62 SCachePage* end=page+aPages;
65 for (;page<end;++page)
67 page->iPage[0].iOwner=NULL;
68 page->iPage[0].iChange=EPageNoChange;
69 iFree.AddFirst(page->iPage[1]);
73 EXPORT_C CPageCache::~CPageCache()
76 All cache pages are deleted. */
79 for (SCachePage* page=iPages,*end=iEnd;page<end;++page)
80 __ASSERT_DEBUG(page->iPage[0].iOwner==NULL,Panic(EPageCacheInUse));
85 TCachePage* CPageCache::Find(TCachePagePool* aPool,TPageRef aRef)
87 // Find a page in the cache.
90 for (SCachePage* page=iPages,*end=iEnd;page<end;++page)
92 if (page->iPage[0].iOwner!=aPool)
94 __ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant());
95 // no other page pool can have locked pages
99 if (page->iPage[0].iRef==aRef)
101 #if defined(__PAGE_CACHE_STATS)
102 if (page->iPage[1].IsQued())
103 iStats.Hit(); // finding a locked page is not a hit, I'm afraid
105 return &page->iPage[1];
108 #if defined(__PAGE_CACHE_STATS)
114 EXPORT_C TPageAbandonFunction TCachePagePool::AcquireL()
115 /** Returns a function that abandons all locked pages for this page pool.
117 @return A function that abandons all locked pages for this page pool. */
122 EXPORT_C TAny* TCachePagePool::AllocL()
123 /** Allocates an unassigned page.
125 @return Newly allocated page. */
127 CPageCache& cache=*iCache;
128 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
129 TCachePage* page=DoAllocL(cache);
130 page[-1].iOwner=this;
131 __ASSERT_DEBUG(page[-1].iChange==EPageNoChange,User::Invariant());
132 page[-1].iRef=KNullPageRef;
137 EXPORT_C TAny* TCachePagePool::LockL(TPageRef aRef)
138 /** Locks a page and returns a pointer to it.
140 @param aRef Reference to page to lock
141 @return Locked page */
143 if (aRef==KNullPageRef)
144 __LEAVE(KErrNotFound);
146 CPageCache& cache=*iCache;
147 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
148 TCachePage* page=cache.Find(this,aRef);
150 { // it's not in the cache, allocate a free page
151 page=DoAllocL(cache);
152 page[-1].iOwner=NULL;
153 __ASSERT_DEBUG(page[-1].iChange==EPageNoChange,User::Invariant());
155 page[-1].iOwner=this;
158 __ASSERT_DEBUG(page->IsQued(),Panic(EPageDoubleLock)); // it mustn't be locked already
163 EXPORT_C TPageRef TCachePagePool::AssignL(const TAny* aPage,TPageReclamation aReclamation)
164 /** Assigns a reference to a new page and unlocks it.
166 @param aPage Page to assign
167 @param aReclamation Flag for page reclaimation settings
168 @return Reference to page */
170 TCachePage* page=(TCachePage*)aPage;
171 __ASSERT_DEBUG(page!=NULL&&!page->IsQued(),Panic(EPageNotLocked));
172 __ASSERT_DEBUG(page[-1].iOwner==this&&page[-1].iRef==KNullPageRef,User::Invariant());
173 CPageCache& cache=*iCache;
174 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
175 page[-1].iOwner=NULL;
176 __ASSERT_DEBUG(page[-1].iChange==EPageNoChange,User::Invariant());
177 cache.iFree.AddLast(*page);
178 TPageRef ref=ExtendL(page,aReclamation);
179 page[-1].iOwner=this;
184 EXPORT_C void TCachePagePool::UpdateL(const TAny* aPage)
187 @param aPage Page to update */
189 TCachePage* page=(TCachePage*)aPage;
190 __ASSERT_DEBUG(page!=NULL&&!page->IsQued(),Panic(EPageNotLocked));
191 __ASSERT_DEBUG(page[-1].iOwner==this&&page[-1].iRef!=KNullPageRef,User::Invariant());
192 CPageCache& cache=*iCache;
193 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
194 page[-1].iOwner=NULL;
195 page[-1].iChange=EPageNoChange;
196 cache.iFree.AddLast(*page);
197 WriteL(page[-1].iRef,page,EPageUpdate);
198 page[-1].iOwner=this;
201 EXPORT_C void TCachePagePool::Unlock(const TAny* aPage,TPageChange aChange)
203 // Unlock a page, depending on aChange:
204 // EPageNoChange: just unlock
205 // EPageDirty: mark the page as dirty
206 // EPageUpdate: mark the page as needing a safe update
207 // EPageAbandon: discard the page
211 @param aPage Page to unlock
212 @param aChange How the page should be treated once it is unlocked */
214 TCachePage* page=(TCachePage*)aPage;
215 __ASSERT_DEBUG(page!=NULL&&!page->IsQued(),Panic(EPageNotLocked));
216 __ASSERT_DEBUG(page[-1].iOwner==this&&page[-1].iRef!=KNullPageRef,User::Invariant());
217 CPageCache& cache=*iCache;
218 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
219 if (aChange>page[-1].iChange)
220 page[-1].iChange=aChange;
221 else if (aChange<EPageNoChange)
223 page[-1].iOwner=NULL;
224 page[-1].iChange=EPageNoChange;
226 cache.iFree.AddLast(*page);
229 EXPORT_C TInt TCachePagePool::Flush()
230 /** Flush all pages in this pool from the cache.
232 It ensures that any dirty pages are written into persistent storage, but does
233 not remove them from the cache.
235 @return KErrNone if successful, otherwise another of the system-wide error
242 EXPORT_C void TCachePagePool::FlushL()
243 /** Flushes all pages in this pool from the cache, leaving with a system-wide error
244 code if an error occurs. */
246 CPageCache& cache=*iCache;
247 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
248 for (SCachePage* page=cache.iPages,*end=cache.iEnd;page<end;++page)
250 __ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant()); // there mustn't be any locked pages
251 if (page->iPage[0].iOwner!=this)
254 TPageChange change=page->iPage[0].iChange;
255 if (change>EPageNoChange)
257 WriteL(page->iPage[0].iRef,&page->iPage[1],change);
258 page->iPage[0].iChange=EPageNoChange;
263 EXPORT_C void TCachePagePool::Purge()
264 /** Purge all pages in this pool from the cache.
266 This discards all pages from the cache, without saving dirty pages. */
268 CPageCache& cache=*iCache;
269 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
270 for (SCachePage* page=cache.iPages,*end=cache.iEnd;page<end;++page)
272 __ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant()); // there mustn't be any locked pages
273 if (page->iPage[0].iOwner!=this)
276 page->iPage[0].iOwner=NULL;
277 page->iPage[0].iChange=EPageNoChange;
278 page->iPage[1].Link().Deque();
279 cache.iFree.AddFirst(page->iPage[1]);
283 EXPORT_C void TCachePagePool::DoDeleteL(TPageRef aRef)
285 // Default implementation removing from cache.
288 __ASSERT_DEBUG(aRef!=KNullPageRef,User::Invariant());
289 CPageCache& cache=*iCache;
290 __ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
291 TCachePage* page=cache.Find(this,aRef);
294 page[-1].iOwner=NULL;
295 page[-1].iChange=EPageNoChange;
296 page->Link().Deque();
297 cache.iFree.AddFirst(*page);
301 void TCachePagePool::DoAbandon(MPagePool& aPool)
303 // Abandon all locked pages in this page pool.
306 TCachePagePool& pool=STATIC_CAST(TCachePagePool&,aPool);
307 CPageCache& cache=*pool.iCache;
308 __ASSERT_DEBUG(pool.iCache!=NULL,Panic(EPageNotOpen));
309 for (SCachePage* page=cache.iPages,*end=cache.iEnd;page<end;++page)
311 if (page->iPage[1].IsQued())
314 __ASSERT_DEBUG(page->iPage[0].iOwner==&pool,User::Invariant());
315 // no other page pool can have locked pages
316 page->iPage[0].iOwner=NULL;
317 page->iPage[0].iChange=EPageNoChange;
318 cache.iFree.AddFirst(page->iPage[1]);
322 TCachePage* TCachePagePool::DoAllocL(CPageCache& aCache)
324 __ASSERT_ALWAYS(!aCache.iFree.IsEmpty(),Panic(EPageCacheFull));
325 // too small a cache or pages left locked unnecessarily
326 TCachePage* page=aCache.iFree.First();
327 TPageChange change=page[-1].iChange;
328 if (change>EPageNoChange)
330 page[-1].iOwner->WriteL(page[-1].iRef,page,change);
331 page[-1].iChange=EPageNoChange;