os/persistentdata/persistentstorage/store/UPAGE/UP_CACHE.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "UP_STD.H"
    17 
    18 struct SCachePage
    19 	{
    20 	TCachePage iPage[1];
    21 	TUint8 iData[KPoolPageSize];
    22 	};
    23 
    24 EXPORT_C CPageCache* CPageCache::NewL(TInt aPages)
    25 /** Allocates and constructs a new page cache.
    26 
    27 @param aPages Number of pages in the cache
    28 @return New page cache object */
    29 	{
    30 	CPageCache* cache=CPageCache::NewLC(aPages);
    31 	CleanupStack::Pop();
    32 	return cache;
    33 	}
    34 
    35 EXPORT_C CPageCache* CPageCache::NewLC(TInt aPages)
    36 /** Allocates and constructs a new page cache, and leaves it on the cleanup stack.
    37 
    38 @param aPages Number of pages in the cache
    39 @return New page cache object */
    40 	{
    41 	CPageCache* cache=new(ELeave) CPageCache;
    42 	CleanupStack::PushL(cache);
    43 	cache->ConstructL(aPages);
    44 	return cache;
    45 	}
    46 
    47 EXPORT_C CPageCache::CPageCache()
    48 	: iFree(_FOFF(TCachePage,iLink)-_FOFF(SCachePage,iPage[1]))/*,iPages(NULL),iEnd(NULL)*/
    49 /** Default constructor. */
    50 	{
    51 #if defined(__PAGE_CACHE_STATS)
    52 //	iStats.Reset();
    53 #endif
    54 	}
    55 
    56 EXPORT_C void CPageCache::ConstructL(TInt aPages)
    57 /** Second phase construction.
    58 
    59 @param aPages Number of pages in the cache */
    60 	{
    61 	SCachePage* page=new(ELeave) SCachePage[aPages];
    62 	SCachePage* end=page+aPages;
    63 	iPages=page;
    64 	iEnd=end;
    65 	for (;page<end;++page)
    66 		{
    67 		page->iPage[0].iOwner=NULL;
    68 		page->iPage[0].iChange=EPageNoChange;
    69 		iFree.AddFirst(page->iPage[1]);
    70 		}
    71 	}
    72 
    73 EXPORT_C CPageCache::~CPageCache()
    74 /** Destructor.
    75 
    76 All cache pages are deleted. */
    77 	{
    78 #if defined (_DEBUG)
    79 	for (SCachePage* page=iPages,*end=iEnd;page<end;++page)
    80 		__ASSERT_DEBUG(page->iPage[0].iOwner==NULL,Panic(EPageCacheInUse));
    81 #endif
    82 	delete[] iPages;
    83 	}
    84 
    85 TCachePage* CPageCache::Find(TCachePagePool* aPool,TPageRef aRef)
    86 //
    87 // Find a page in the cache.
    88 //
    89 	{
    90 	for (SCachePage* page=iPages,*end=iEnd;page<end;++page)
    91 		{
    92 		if (page->iPage[0].iOwner!=aPool)
    93 			{
    94 			__ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant());
    95 				// no other page pool can have locked pages
    96 			continue;
    97 			}
    98 //
    99 		if (page->iPage[0].iRef==aRef)
   100 			{
   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
   104 #endif
   105 			return &page->iPage[1];
   106 			}
   107 		}
   108 #if defined(__PAGE_CACHE_STATS)
   109 	iStats.Miss();
   110 #endif
   111 	return NULL;
   112 	}
   113 
   114 EXPORT_C TPageAbandonFunction TCachePagePool::AcquireL()
   115 /** Returns a function that abandons all locked pages for this page pool. 
   116 
   117 @return A function that abandons all locked pages for this page pool. */
   118 	{
   119 	return &DoAbandon;
   120 	}
   121 
   122 EXPORT_C TAny* TCachePagePool::AllocL()
   123 /** Allocates an unassigned page. 
   124 
   125 @return Newly allocated page. */
   126 	{
   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;
   133 	page->Deque();
   134 	return page;
   135 	}
   136 
   137 EXPORT_C TAny* TCachePagePool::LockL(TPageRef aRef)
   138 /** Locks a page and returns a pointer to it.
   139 
   140 @param aRef Reference to page to lock
   141 @return Locked page */
   142 	{
   143 	if (aRef==KNullPageRef)
   144 		__LEAVE(KErrNotFound);
   145 //
   146 	CPageCache& cache=*iCache;
   147 	__ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
   148 	TCachePage* page=cache.Find(this,aRef);
   149 	if (page==NULL)
   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());
   154 		ReadL(aRef,page);
   155 		page[-1].iOwner=this;
   156 		page[-1].iRef=aRef;
   157 		}
   158 	__ASSERT_DEBUG(page->IsQued(),Panic(EPageDoubleLock)); // it mustn't be locked already
   159 	page->Deque();
   160 	return page;
   161 	}
   162 
   163 EXPORT_C TPageRef TCachePagePool::AssignL(const TAny* aPage,TPageReclamation aReclamation)
   164 /** Assigns a reference to a new page and unlocks it. 
   165 
   166 @param aPage Page to assign
   167 @param aReclamation Flag for page reclaimation settings
   168 @return Reference to page */
   169 	{
   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;
   180 	page[-1].iRef=ref;
   181 	return ref;
   182 	}
   183 
   184 EXPORT_C void TCachePagePool::UpdateL(const TAny* aPage)
   185 /** Updates a page. 
   186 
   187 @param aPage Page to update */
   188 	{
   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;
   199 	}
   200 
   201 EXPORT_C void TCachePagePool::Unlock(const TAny* aPage,TPageChange aChange)
   202 //
   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
   208 //
   209 /** Unlocks a page.
   210 
   211 @param aPage Page to unlock
   212 @param aChange How the page should be treated once it is unlocked */
   213 	{
   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)
   222 		{
   223 		page[-1].iOwner=NULL;
   224 		page[-1].iChange=EPageNoChange;
   225 		}
   226 	cache.iFree.AddLast(*page);
   227 	}
   228 
   229 EXPORT_C TInt TCachePagePool::Flush()
   230 /** Flush all pages in this pool from the cache.
   231 
   232 It ensures that any dirty pages are written into persistent storage, but does 
   233 not remove them from the cache.
   234 
   235 @return KErrNone if successful, otherwise another of the system-wide error 
   236 codes. */
   237 	{
   238 	TRAPD(r,FlushL());
   239 	return r;
   240 	}
   241 
   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. */
   245 	{
   246 	CPageCache& cache=*iCache;
   247 	__ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
   248 	for (SCachePage* page=cache.iPages,*end=cache.iEnd;page<end;++page)
   249 		{
   250 		__ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant()); // there mustn't be any locked pages
   251 		if (page->iPage[0].iOwner!=this)
   252 			continue;
   253 //
   254 		TPageChange change=page->iPage[0].iChange;
   255 		if (change>EPageNoChange)
   256 			{
   257 			WriteL(page->iPage[0].iRef,&page->iPage[1],change);
   258 			page->iPage[0].iChange=EPageNoChange;
   259 			}
   260 		}
   261 	}
   262 
   263 EXPORT_C void TCachePagePool::Purge()
   264 /** Purge all pages in this pool from the cache.
   265 
   266 This discards all pages from the cache, without saving dirty pages. */
   267 	{
   268 	CPageCache& cache=*iCache;
   269 	__ASSERT_DEBUG(iCache!=NULL,Panic(EPageNotOpen));
   270 	for (SCachePage* page=cache.iPages,*end=cache.iEnd;page<end;++page)
   271 		{
   272 		__ASSERT_DEBUG(page->iPage[1].IsQued(),User::Invariant()); // there mustn't be any locked pages
   273 		if (page->iPage[0].iOwner!=this)
   274 			continue;
   275 //
   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]);
   280 		}
   281 	}
   282 
   283 EXPORT_C void TCachePagePool::DoDeleteL(TPageRef aRef)
   284 //
   285 // Default implementation removing from cache.
   286 //
   287 	{
   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);
   292 	if (page!=NULL)
   293 		{
   294 		page[-1].iOwner=NULL;
   295 		page[-1].iChange=EPageNoChange;
   296 		page->Link().Deque();
   297 		cache.iFree.AddFirst(*page);
   298 		}
   299 	}
   300 
   301 void TCachePagePool::DoAbandon(MPagePool& aPool)
   302 //
   303 // Abandon all locked pages in this page pool.
   304 //
   305 	{
   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)
   310 		{
   311 		if (page->iPage[1].IsQued())
   312 			continue;
   313 //
   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]);
   319 		}
   320 	}
   321 
   322 TCachePage* TCachePagePool::DoAllocL(CPageCache& aCache)
   323 	{
   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)
   329 		{
   330 		page[-1].iOwner->WriteL(page[-1].iRef,page,change);
   331 		page[-1].iChange=EPageNoChange;
   332 		}
   333 	return page;
   334 	}
   335