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 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include "UT_STD.H"
|
sl@0
|
17 |
|
sl@0
|
18 |
// PFS frame-related utilities
|
sl@0
|
19 |
|
sl@0
|
20 |
TInt SkipLink(TInt anExtent)
|
sl@0
|
21 |
//
|
sl@0
|
22 |
// anExtent is the end-of-stream, where is the next link pos?
|
sl@0
|
23 |
// : add the size of the next frame link, or skip past anchor link
|
sl@0
|
24 |
//
|
sl@0
|
25 |
{return anExtent+Min(KSizeFrameDes16,(-anExtent)&KMaskFrameLength16);}
|
sl@0
|
26 |
|
sl@0
|
27 |
inline TBool SpaceFor(TInt aSpace,TInt anEnd,TInt aLength)
|
sl@0
|
28 |
// Check if there is space for at least aLength between links at aSpace and anEnd
|
sl@0
|
29 |
{return SkipLink(aSpace+aLength)<=anEnd;}
|
sl@0
|
30 |
|
sl@0
|
31 |
TBool Fits(TInt aSpace,TInt anEnd,TInt aLength)
|
sl@0
|
32 |
//
|
sl@0
|
33 |
// Check that aLength can be relocated into the space
|
sl@0
|
34 |
// either an exact fit, or leaves space for a free frame of at least one byte
|
sl@0
|
35 |
//
|
sl@0
|
36 |
{
|
sl@0
|
37 |
TInt end=SkipLink(aSpace+aLength);
|
sl@0
|
38 |
return end==anEnd ? ETrue : SpaceFor(end,anEnd,1);
|
sl@0
|
39 |
}
|
sl@0
|
40 |
|
sl@0
|
41 |
TInt ExtentL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase,TInt aStream)
|
sl@0
|
42 |
//
|
sl@0
|
43 |
// scan a stream to discover it's extent
|
sl@0
|
44 |
//
|
sl@0
|
45 |
{
|
sl@0
|
46 |
__ASSERT_DEBUG(aMark.RelatesTo(aHost),User::Invariant());
|
sl@0
|
47 |
__ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
|
sl@0
|
48 |
__ASSERT_DEBUG(aStream>=0,User::Invariant());
|
sl@0
|
49 |
//
|
sl@0
|
50 |
aMark.SeekL(aHost,RFrame16Buf::Position(aBase,aStream)-KSizeFrameDes16);
|
sl@0
|
51 |
TFrameDes16 des;
|
sl@0
|
52 |
aMark.ReadL(aHost,&des,KSizeFrameDes16);
|
sl@0
|
53 |
TInt frame=des;
|
sl@0
|
54 |
if ((frame&KMaskFrameType16)!=EFrameData16)
|
sl@0
|
55 |
if ((frame&KMaskFrameType16)!=EFrameDescriptive16)
|
sl@0
|
56 |
{
|
sl@0
|
57 |
aMark.Withdraw(aHost);
|
sl@0
|
58 |
__LEAVE(KErrCorrupt);
|
sl@0
|
59 |
}
|
sl@0
|
60 |
//
|
sl@0
|
61 |
TInt anchor=((aStream>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
|
sl@0
|
62 |
do
|
sl@0
|
63 |
{
|
sl@0
|
64 |
frame&=KMaskFrameLength16;
|
sl@0
|
65 |
TInt lim=anchor-aStream;
|
sl@0
|
66 |
if (frame!=KFrameOpen16)
|
sl@0
|
67 |
{
|
sl@0
|
68 |
if (frame>=lim)
|
sl@0
|
69 |
{
|
sl@0
|
70 |
aMark.Withdraw(aHost);
|
sl@0
|
71 |
__LEAVE(KErrCorrupt);
|
sl@0
|
72 |
}
|
sl@0
|
73 |
return aStream+frame;
|
sl@0
|
74 |
}
|
sl@0
|
75 |
aMark.SeekL(aHost,EStreamMark,lim);
|
sl@0
|
76 |
aStream=anchor;
|
sl@0
|
77 |
anchor+=KFrameFullLength16;
|
sl@0
|
78 |
aMark.ReadL(aHost,&des,KSizeFrameDes16);
|
sl@0
|
79 |
frame=des;
|
sl@0
|
80 |
} while ((frame&KMaskFrameType16)==EFrameContinuation16);
|
sl@0
|
81 |
return aStream;
|
sl@0
|
82 |
}
|
sl@0
|
83 |
|
sl@0
|
84 |
|
sl@0
|
85 |
|
sl@0
|
86 |
// Class TRelocatorInput
|
sl@0
|
87 |
// Used to transfer a frame-set within the file
|
sl@0
|
88 |
|
sl@0
|
89 |
const TInt KRelocatorBufferSize=0xc00;
|
sl@0
|
90 |
|
sl@0
|
91 |
NONSHARABLE_CLASS(TRelocatorInput) : public MStreamInput
|
sl@0
|
92 |
{
|
sl@0
|
93 |
public:
|
sl@0
|
94 |
inline TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark);
|
sl@0
|
95 |
void OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator);
|
sl@0
|
96 |
void CommitL();
|
sl@0
|
97 |
private:
|
sl@0
|
98 |
TInt PushL(const TAny* aPtr,TInt aMaxLength);
|
sl@0
|
99 |
TStreamTransfer ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer);
|
sl@0
|
100 |
void DesL(TFrameType16 aType);
|
sl@0
|
101 |
//
|
sl@0
|
102 |
inline TStreamExchange& Host() const;
|
sl@0
|
103 |
inline TStreamMark& Mark();
|
sl@0
|
104 |
private:
|
sl@0
|
105 |
TStreamExchange* iHost;
|
sl@0
|
106 |
TStreamMark* iMark;
|
sl@0
|
107 |
TFrameType16 iType;
|
sl@0
|
108 |
TInt iRemain;
|
sl@0
|
109 |
TInt iAnchor;
|
sl@0
|
110 |
TInt iTerminator;
|
sl@0
|
111 |
};
|
sl@0
|
112 |
|
sl@0
|
113 |
inline TRelocatorInput::TRelocatorInput(TStreamExchange& aHost,TStreamMark& aMark)
|
sl@0
|
114 |
: iHost(&aHost),iMark(&aMark)
|
sl@0
|
115 |
{}
|
sl@0
|
116 |
inline TStreamExchange& TRelocatorInput::Host() const
|
sl@0
|
117 |
{
|
sl@0
|
118 |
__ASSERT_DEBUG(iHost!=NULL,User::Invariant());
|
sl@0
|
119 |
return *iHost;
|
sl@0
|
120 |
}
|
sl@0
|
121 |
inline TStreamMark& TRelocatorInput::Mark()
|
sl@0
|
122 |
{
|
sl@0
|
123 |
__ASSERT_DEBUG(iMark!=NULL,User::Invariant());
|
sl@0
|
124 |
return *iMark;
|
sl@0
|
125 |
}
|
sl@0
|
126 |
|
sl@0
|
127 |
void TRelocatorInput::OpenL(TFrameType16 aType,TStreamPos aBase,TInt aPos,TInt aLength,TInt aTerminator)
|
sl@0
|
128 |
//
|
sl@0
|
129 |
// initiate the relocated stream
|
sl@0
|
130 |
//
|
sl@0
|
131 |
{
|
sl@0
|
132 |
__ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
|
sl@0
|
133 |
__ASSERT_DEBUG(aBase>=KStreamBeginning,User::Invariant());
|
sl@0
|
134 |
__ASSERT_DEBUG(aPos>=0,User::Invariant());
|
sl@0
|
135 |
__ASSERT_DEBUG(aLength>0,User::Invariant());
|
sl@0
|
136 |
//
|
sl@0
|
137 |
iType=aType;
|
sl@0
|
138 |
iRemain=aLength;
|
sl@0
|
139 |
iTerminator=aTerminator;
|
sl@0
|
140 |
iAnchor=KFrameFullLength16-(aPos&KMaskFrameLength16);
|
sl@0
|
141 |
Mark().SeekL(Host(),RFrame16Buf::Position(aBase,aPos)-KSizeFrameDes16);
|
sl@0
|
142 |
}
|
sl@0
|
143 |
|
sl@0
|
144 |
|
sl@0
|
145 |
void TRelocatorInput::CommitL()
|
sl@0
|
146 |
//
|
sl@0
|
147 |
// complete the relocated stream
|
sl@0
|
148 |
//
|
sl@0
|
149 |
{
|
sl@0
|
150 |
__ASSERT_DEBUG(iRemain==0,User::Invariant());
|
sl@0
|
151 |
__ASSERT_DEBUG(iAnchor>=0,User::Invariant());
|
sl@0
|
152 |
//
|
sl@0
|
153 |
if (iTerminator<0)
|
sl@0
|
154 |
return;
|
sl@0
|
155 |
//
|
sl@0
|
156 |
TFrameDes16 des[2];
|
sl@0
|
157 |
des[1]=TFrameDes16(iTerminator);
|
sl@0
|
158 |
TInt l=KSizeFrameDes16;
|
sl@0
|
159 |
if (iAnchor<=KSizeFrameDes16)
|
sl@0
|
160 |
l+=iAnchor;
|
sl@0
|
161 |
Mark().WriteL(Host(),(const TUint8*)&des[2]-l,l);
|
sl@0
|
162 |
}
|
sl@0
|
163 |
|
sl@0
|
164 |
TInt TRelocatorInput::PushL(const TAny*,TInt)
|
sl@0
|
165 |
//
|
sl@0
|
166 |
// use a passive write through to the host
|
sl@0
|
167 |
//
|
sl@0
|
168 |
{
|
sl@0
|
169 |
return 0;
|
sl@0
|
170 |
}
|
sl@0
|
171 |
|
sl@0
|
172 |
void TRelocatorInput::DesL(TFrameType16 aType)
|
sl@0
|
173 |
//
|
sl@0
|
174 |
// Write the next frame descriptor
|
sl@0
|
175 |
//
|
sl@0
|
176 |
{
|
sl@0
|
177 |
__ASSERT_DEBUG(aType!=EFrameFree16,User::Invariant());
|
sl@0
|
178 |
TFrameDes16 des=TFrameDes16(iRemain<iAnchor ? aType+iRemain : aType+KFrameOpen16);
|
sl@0
|
179 |
Mark().WriteL(Host(),&des,KSizeFrameDes16);
|
sl@0
|
180 |
}
|
sl@0
|
181 |
|
sl@0
|
182 |
TStreamTransfer TRelocatorInput::ReadFromL(MStreamBuf& aSource,TStreamTransfer aTransfer)
|
sl@0
|
183 |
//
|
sl@0
|
184 |
// bulk of the transfer happens here
|
sl@0
|
185 |
//
|
sl@0
|
186 |
{
|
sl@0
|
187 |
__ASSERT_DEBUG(!(aTransfer>iRemain),User::Invariant());
|
sl@0
|
188 |
do
|
sl@0
|
189 |
{
|
sl@0
|
190 |
TUint8 buf[KRelocatorBufferSize];
|
sl@0
|
191 |
TInt len=aSource.ReadL(buf,aTransfer[sizeof(buf)]);
|
sl@0
|
192 |
if (len==0)
|
sl@0
|
193 |
break;
|
sl@0
|
194 |
aTransfer-=len;
|
sl@0
|
195 |
const TUint8* p=buf;
|
sl@0
|
196 |
if (iType!=EFrameFree16)
|
sl@0
|
197 |
{
|
sl@0
|
198 |
DesL(iType);
|
sl@0
|
199 |
iType=EFrameFree16;
|
sl@0
|
200 |
}
|
sl@0
|
201 |
for (;;)
|
sl@0
|
202 |
{
|
sl@0
|
203 |
if (iAnchor==0)
|
sl@0
|
204 |
{ // write next anchor
|
sl@0
|
205 |
iAnchor=KFrameFullLength16;
|
sl@0
|
206 |
DesL(EFrameContinuation16);
|
sl@0
|
207 |
}
|
sl@0
|
208 |
TInt frame=Min(len,iAnchor);
|
sl@0
|
209 |
Mark().WriteL(Host(),p,frame);
|
sl@0
|
210 |
iAnchor-=frame;
|
sl@0
|
211 |
iRemain-=frame;
|
sl@0
|
212 |
len-=frame;
|
sl@0
|
213 |
if (len==0)
|
sl@0
|
214 |
break;
|
sl@0
|
215 |
p+=frame;
|
sl@0
|
216 |
}
|
sl@0
|
217 |
} while (aTransfer>0);
|
sl@0
|
218 |
return aTransfer;
|
sl@0
|
219 |
}
|
sl@0
|
220 |
|
sl@0
|
221 |
|
sl@0
|
222 |
|
sl@0
|
223 |
// Class TPermanentStoreRelocator
|
sl@0
|
224 |
// used to locate streams for relocation in limited memory
|
sl@0
|
225 |
|
sl@0
|
226 |
class TPermanentStoreRelocator
|
sl@0
|
227 |
{
|
sl@0
|
228 |
#if defined(__SMALL_BUNDLE)
|
sl@0
|
229 |
enum {EBundleSize=8-1};
|
sl@0
|
230 |
#else
|
sl@0
|
231 |
enum {EBundleSize=64-1};
|
sl@0
|
232 |
#endif
|
sl@0
|
233 |
public:
|
sl@0
|
234 |
typedef CPermanentStoreCollector::TEntry TEntry;
|
sl@0
|
235 |
public:
|
sl@0
|
236 |
void Reset();
|
sl@0
|
237 |
TBool Reset(TInt aPos);
|
sl@0
|
238 |
TInt FillL(CPermanentStoreToc& aToc);
|
sl@0
|
239 |
void EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase);
|
sl@0
|
240 |
//
|
sl@0
|
241 |
TBool FindFit(TInt aSpace,TInt anEnd);
|
sl@0
|
242 |
inline const TEntry* Current() const;
|
sl@0
|
243 |
void Relocated(const TEntry* anEntry);
|
sl@0
|
244 |
//
|
sl@0
|
245 |
TInt Extent(TInt aStream) const;
|
sl@0
|
246 |
inline TInt MinLength() const;
|
sl@0
|
247 |
private:
|
sl@0
|
248 |
static void Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue);
|
sl@0
|
249 |
static TEntry PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue);
|
sl@0
|
250 |
private:
|
sl@0
|
251 |
TEntry* iNext;
|
sl@0
|
252 |
const TEntry* iFinish;
|
sl@0
|
253 |
TInt iMore;
|
sl@0
|
254 |
TInt iPos;
|
sl@0
|
255 |
TInt iCurrentMin,iMinLength;
|
sl@0
|
256 |
TEntry iTable[EBundleSize];
|
sl@0
|
257 |
};
|
sl@0
|
258 |
|
sl@0
|
259 |
inline const TPermanentStoreRelocator::TEntry* TPermanentStoreRelocator::Current() const
|
sl@0
|
260 |
{__ASSERT_DEBUG(iNext>=iTable&&iNext<iFinish,User::Invariant());return iNext;}
|
sl@0
|
261 |
inline TInt TPermanentStoreRelocator::MinLength() const
|
sl@0
|
262 |
{return iMinLength;}
|
sl@0
|
263 |
|
sl@0
|
264 |
void TPermanentStoreRelocator::Reset()
|
sl@0
|
265 |
//
|
sl@0
|
266 |
// reset the cached length values
|
sl@0
|
267 |
//
|
sl@0
|
268 |
{
|
sl@0
|
269 |
iPos=0;
|
sl@0
|
270 |
iMinLength=1;
|
sl@0
|
271 |
}
|
sl@0
|
272 |
|
sl@0
|
273 |
TBool TPermanentStoreRelocator::Reset(TInt aPos)
|
sl@0
|
274 |
//
|
sl@0
|
275 |
// reset the iterator for a new scan from aPos
|
sl@0
|
276 |
//
|
sl@0
|
277 |
{
|
sl@0
|
278 |
__ASSERT_DEBUG(aPos>=0,User::Invariant());
|
sl@0
|
279 |
//
|
sl@0
|
280 |
iCurrentMin=KMaxTInt;
|
sl@0
|
281 |
TEntry* e=iTable;
|
sl@0
|
282 |
if (aPos>=iPos || e==iFinish || aPos<e->entry.ref)
|
sl@0
|
283 |
{ // rescan required
|
sl@0
|
284 |
iFinish=iNext=NULL;
|
sl@0
|
285 |
iMore=-1;
|
sl@0
|
286 |
iPos=aPos;
|
sl@0
|
287 |
return EFalse;
|
sl@0
|
288 |
}
|
sl@0
|
289 |
|
sl@0
|
290 |
// can use current data
|
sl@0
|
291 |
for (;e->entry.ref!=aPos;++e)
|
sl@0
|
292 |
{
|
sl@0
|
293 |
__ASSERT_DEBUG(e->entry.ref<aPos,User::Invariant());
|
sl@0
|
294 |
__ASSERT_DEBUG(e<iFinish,User::Invariant());
|
sl@0
|
295 |
}
|
sl@0
|
296 |
__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
|
sl@0
|
297 |
iNext=e;
|
sl@0
|
298 |
return ETrue;
|
sl@0
|
299 |
}
|
sl@0
|
300 |
|
sl@0
|
301 |
TBool TPermanentStoreRelocator::FillL(CPermanentStoreToc& aToc)
|
sl@0
|
302 |
//
|
sl@0
|
303 |
// Fill the table with the next bundle of stream offsets
|
sl@0
|
304 |
// return if there are more available
|
sl@0
|
305 |
//
|
sl@0
|
306 |
{
|
sl@0
|
307 |
__ASSERT_DEBUG(iNext==iFinish,User::Invariant());
|
sl@0
|
308 |
if (!iMore)
|
sl@0
|
309 |
return EFalse;
|
sl@0
|
310 |
//
|
sl@0
|
311 |
__ASSERT_DEBUG(iPos>=0,User::Invariant());
|
sl@0
|
312 |
//
|
sl@0
|
313 |
RPermanentStoreTocIter iter(aToc);
|
sl@0
|
314 |
CleanupReleasePushL(iter);
|
sl@0
|
315 |
|
sl@0
|
316 |
TInt streams=0;
|
sl@0
|
317 |
TInt from=iPos;
|
sl@0
|
318 |
TEntry* table=iTable;
|
sl@0
|
319 |
TEntry* last=table;
|
sl@0
|
320 |
TEntry* end=table+EBundleSize;
|
sl@0
|
321 |
TEntry entry;
|
sl@0
|
322 |
__DEBUG(entry.len=-1);
|
sl@0
|
323 |
for (iter.ResetL();iter.NextL(entry.entry);)
|
sl@0
|
324 |
{
|
sl@0
|
325 |
if (entry.entry.handle<0)
|
sl@0
|
326 |
continue;
|
sl@0
|
327 |
TInt off=entry.entry.ref;
|
sl@0
|
328 |
if (off<from)
|
sl@0
|
329 |
continue;
|
sl@0
|
330 |
++streams;
|
sl@0
|
331 |
if (last<end)
|
sl@0
|
332 |
Push(table,last++,entry); // push onto the bottom
|
sl@0
|
333 |
else if (off<table->entry.ref)
|
sl@0
|
334 |
PopPush(table,last,entry); // replace item in heap
|
sl@0
|
335 |
}
|
sl@0
|
336 |
CleanupStack::PopAndDestroy();
|
sl@0
|
337 |
|
sl@0
|
338 |
iMore=streams+table-last;
|
sl@0
|
339 |
if (streams)
|
sl@0
|
340 |
iPos=table->entry.ref+1; // largest offset in table
|
sl@0
|
341 |
iNext=table;
|
sl@0
|
342 |
iFinish=last;
|
sl@0
|
343 |
while (--last>table)
|
sl@0
|
344 |
*last=PopPush(table,last,*last);
|
sl@0
|
345 |
return streams>0;
|
sl@0
|
346 |
}
|
sl@0
|
347 |
|
sl@0
|
348 |
void TPermanentStoreRelocator::Push(TEntry* aHeap,TEntry* aHole,const TEntry& aValue)
|
sl@0
|
349 |
//
|
sl@0
|
350 |
// push aValue onto the heap (which will grow)
|
sl@0
|
351 |
//
|
sl@0
|
352 |
{
|
sl@0
|
353 |
TInt ix=aHole+1-aHeap;
|
sl@0
|
354 |
while (aHole>aHeap)
|
sl@0
|
355 |
{
|
sl@0
|
356 |
TInt link=ix-(ix>>1);
|
sl@0
|
357 |
ix-=link;
|
sl@0
|
358 |
TEntry* parent=aHole-link;
|
sl@0
|
359 |
if (parent->entry.ref>=aValue.entry.ref)
|
sl@0
|
360 |
break;
|
sl@0
|
361 |
*aHole=*parent;
|
sl@0
|
362 |
aHole=parent;
|
sl@0
|
363 |
}
|
sl@0
|
364 |
*aHole=aValue;
|
sl@0
|
365 |
}
|
sl@0
|
366 |
|
sl@0
|
367 |
TPermanentStoreRelocator::TEntry TPermanentStoreRelocator::PopPush(TEntry* aHeap,TEntry* anEnd,const TEntry& aValue)
|
sl@0
|
368 |
//
|
sl@0
|
369 |
// pop the top and push aValue
|
sl@0
|
370 |
//
|
sl@0
|
371 |
{
|
sl@0
|
372 |
TEntry top=*aHeap;
|
sl@0
|
373 |
TEntry* hole=aHeap;
|
sl@0
|
374 |
--anEnd;
|
sl@0
|
375 |
TInt ix=1;
|
sl@0
|
376 |
for (;;)
|
sl@0
|
377 |
{
|
sl@0
|
378 |
TEntry* child=hole+ix;
|
sl@0
|
379 |
ix<<=1;
|
sl@0
|
380 |
if (child>anEnd)
|
sl@0
|
381 |
break;
|
sl@0
|
382 |
if (child<anEnd && (child+1)->entry.ref>child->entry.ref)
|
sl@0
|
383 |
{
|
sl@0
|
384 |
++ix;
|
sl@0
|
385 |
++child;
|
sl@0
|
386 |
}
|
sl@0
|
387 |
if (child->entry.ref<=aValue.entry.ref)
|
sl@0
|
388 |
break;
|
sl@0
|
389 |
*hole=*child;
|
sl@0
|
390 |
hole=child;
|
sl@0
|
391 |
}
|
sl@0
|
392 |
*hole=aValue;
|
sl@0
|
393 |
return top;
|
sl@0
|
394 |
}
|
sl@0
|
395 |
|
sl@0
|
396 |
void TPermanentStoreRelocator::EvaluateLengthsL(TStreamExchange& aHost,TStreamMark& aMark,TStreamPos aBase)
|
sl@0
|
397 |
//
|
sl@0
|
398 |
// Evaluate the lengths for all the entries in the table
|
sl@0
|
399 |
//
|
sl@0
|
400 |
{
|
sl@0
|
401 |
const TEntry* end=iFinish;
|
sl@0
|
402 |
for (TEntry* e=iTable;e<end;++e)
|
sl@0
|
403 |
{
|
sl@0
|
404 |
__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
|
sl@0
|
405 |
__ASSERT_DEBUG(e->entry.ref>=0,User::Invariant());
|
sl@0
|
406 |
__ASSERT_DEBUG(e->len==-1,User::Invariant());
|
sl@0
|
407 |
TInt pos=e->entry.ref;
|
sl@0
|
408 |
e->len=ExtentL(aHost,aMark,aBase,pos)-pos;
|
sl@0
|
409 |
}
|
sl@0
|
410 |
}
|
sl@0
|
411 |
|
sl@0
|
412 |
TBool TPermanentStoreRelocator::FindFit(TInt aSpace,TInt anEnd)
|
sl@0
|
413 |
//
|
sl@0
|
414 |
// Find a stream which will fit into the space
|
sl@0
|
415 |
//
|
sl@0
|
416 |
{
|
sl@0
|
417 |
const TEntry* end=iFinish;
|
sl@0
|
418 |
TEntry* e;
|
sl@0
|
419 |
for (e=iNext;e<end;++e)
|
sl@0
|
420 |
{
|
sl@0
|
421 |
if (e->entry.handle<0)
|
sl@0
|
422 |
continue;
|
sl@0
|
423 |
TInt len=e->len;
|
sl@0
|
424 |
__ASSERT_DEBUG(len>0,User::Invariant());
|
sl@0
|
425 |
if (Fits(aSpace,anEnd,len))
|
sl@0
|
426 |
{
|
sl@0
|
427 |
iNext=e;
|
sl@0
|
428 |
return ETrue;
|
sl@0
|
429 |
}
|
sl@0
|
430 |
// len has been left behind, check for minimum
|
sl@0
|
431 |
if (len<iCurrentMin)
|
sl@0
|
432 |
iCurrentMin=len;
|
sl@0
|
433 |
}
|
sl@0
|
434 |
// if no more data, use current min as cached value
|
sl@0
|
435 |
if (iMore==0)
|
sl@0
|
436 |
iMinLength=iCurrentMin;
|
sl@0
|
437 |
iNext=e;
|
sl@0
|
438 |
return EFalse;
|
sl@0
|
439 |
}
|
sl@0
|
440 |
|
sl@0
|
441 |
void TPermanentStoreRelocator::Relocated(const TEntry* anEntry)
|
sl@0
|
442 |
//
|
sl@0
|
443 |
// Relocation has been successful
|
sl@0
|
444 |
//
|
sl@0
|
445 |
{
|
sl@0
|
446 |
__ASSERT_DEBUG(anEntry==iNext,User::Invariant());
|
sl@0
|
447 |
TEntry* e=CONST_CAST(TEntry*,anEntry);
|
sl@0
|
448 |
e->entry.handle=-1;
|
sl@0
|
449 |
iNext=e+1;
|
sl@0
|
450 |
}
|
sl@0
|
451 |
|
sl@0
|
452 |
TInt TPermanentStoreRelocator::Extent(TInt aStream) const
|
sl@0
|
453 |
//
|
sl@0
|
454 |
// Return this stream extent if we know it
|
sl@0
|
455 |
//
|
sl@0
|
456 |
{
|
sl@0
|
457 |
const TEntry* e=iTable;
|
sl@0
|
458 |
if (aStream>=iPos || e==iFinish || aStream<e->entry.ref)
|
sl@0
|
459 |
return -1; // we don't know it
|
sl@0
|
460 |
for (;e->entry.ref!=aStream;++e)
|
sl@0
|
461 |
{
|
sl@0
|
462 |
__ASSERT_DEBUG(e->entry.ref<aStream,User::Invariant());
|
sl@0
|
463 |
__ASSERT_DEBUG(e<iFinish,User::Invariant());
|
sl@0
|
464 |
}
|
sl@0
|
465 |
__ASSERT_DEBUG(e->entry.handle>=0,User::Invariant());
|
sl@0
|
466 |
__ASSERT_DEBUG(e->len>=0,User::Invariant());
|
sl@0
|
467 |
return aStream+e->len;
|
sl@0
|
468 |
}
|
sl@0
|
469 |
|
sl@0
|
470 |
|
sl@0
|
471 |
|
sl@0
|
472 |
// class TPermanentStoreStreamIter
|
sl@0
|
473 |
|
sl@0
|
474 |
void TPermanentStoreStreamIter::Reset()
|
sl@0
|
475 |
//
|
sl@0
|
476 |
// reset the iterator for a new scan
|
sl@0
|
477 |
//
|
sl@0
|
478 |
{
|
sl@0
|
479 |
iNext=NULL;
|
sl@0
|
480 |
iFinish=NULL;
|
sl@0
|
481 |
iMore=-1;
|
sl@0
|
482 |
iPos=0;
|
sl@0
|
483 |
}
|
sl@0
|
484 |
|
sl@0
|
485 |
TInt TPermanentStoreStreamIter::FillL(CPermanentStoreToc& aToc)
|
sl@0
|
486 |
//
|
sl@0
|
487 |
// Fill the table with the next bundle of stream offsets
|
sl@0
|
488 |
// return the total streams left to iterate
|
sl@0
|
489 |
//
|
sl@0
|
490 |
{
|
sl@0
|
491 |
__ASSERT_DEBUG(iNext==iFinish,User::Invariant());
|
sl@0
|
492 |
if (!iMore)
|
sl@0
|
493 |
return 0;
|
sl@0
|
494 |
//
|
sl@0
|
495 |
__ASSERT_DEBUG(iPos>=0,User::Invariant());
|
sl@0
|
496 |
//
|
sl@0
|
497 |
RPermanentStoreTocIter iter(aToc);
|
sl@0
|
498 |
CleanupReleasePushL(iter);
|
sl@0
|
499 |
|
sl@0
|
500 |
TInt streams=0;
|
sl@0
|
501 |
TInt from=iPos;
|
sl@0
|
502 |
TInt* heap=iTable;
|
sl@0
|
503 |
TInt* last=heap;
|
sl@0
|
504 |
TInt* end=heap+EBundleSize;
|
sl@0
|
505 |
RPermanentStoreTocIter::TEntry entry;
|
sl@0
|
506 |
for (iter.ResetL();iter.NextL(entry);)
|
sl@0
|
507 |
{
|
sl@0
|
508 |
if (entry.handle<0)
|
sl@0
|
509 |
continue;
|
sl@0
|
510 |
TInt off=entry.ref;
|
sl@0
|
511 |
if (off<from)
|
sl@0
|
512 |
continue;
|
sl@0
|
513 |
++streams;
|
sl@0
|
514 |
if (last<end)
|
sl@0
|
515 |
Push(heap,last++,off); // push onto the bottom
|
sl@0
|
516 |
else if (off<*heap)
|
sl@0
|
517 |
PopPush(heap,last,off); // replace item in heap
|
sl@0
|
518 |
}
|
sl@0
|
519 |
CleanupStack::PopAndDestroy();
|
sl@0
|
520 |
|
sl@0
|
521 |
iMore=streams+(heap-last);
|
sl@0
|
522 |
iPos=*heap+1; // largest offset in table
|
sl@0
|
523 |
iNext=heap;
|
sl@0
|
524 |
iFinish=last;
|
sl@0
|
525 |
while (--last>heap)
|
sl@0
|
526 |
*last=PopPush(heap,last,*last);
|
sl@0
|
527 |
return streams;
|
sl@0
|
528 |
}
|
sl@0
|
529 |
|
sl@0
|
530 |
TInt TPermanentStoreStreamIter::Next()
|
sl@0
|
531 |
//
|
sl@0
|
532 |
// return the next offset, or <0 if the table is empty
|
sl@0
|
533 |
//
|
sl@0
|
534 |
{
|
sl@0
|
535 |
__ASSERT_DEBUG(iMore>=0,User::Invariant());
|
sl@0
|
536 |
//
|
sl@0
|
537 |
while (iNext<iFinish)
|
sl@0
|
538 |
{
|
sl@0
|
539 |
TInt off=*iNext++;
|
sl@0
|
540 |
if (off>=0)
|
sl@0
|
541 |
return off;
|
sl@0
|
542 |
}
|
sl@0
|
543 |
return -1;
|
sl@0
|
544 |
}
|
sl@0
|
545 |
|
sl@0
|
546 |
void TPermanentStoreStreamIter::Relocated(TInt aStream)
|
sl@0
|
547 |
//
|
sl@0
|
548 |
// aStream has been relocated, mark the table
|
sl@0
|
549 |
//
|
sl@0
|
550 |
{
|
sl@0
|
551 |
__ASSERT_DEBUG(iMore>=0,User::Invariant());
|
sl@0
|
552 |
//
|
sl@0
|
553 |
if (aStream>=iPos)
|
sl@0
|
554 |
return;
|
sl@0
|
555 |
TInt* p=iNext;
|
sl@0
|
556 |
for (;;)
|
sl@0
|
557 |
{
|
sl@0
|
558 |
if (p==iFinish)
|
sl@0
|
559 |
return;
|
sl@0
|
560 |
if (*p>=0)
|
sl@0
|
561 |
break;
|
sl@0
|
562 |
++p;
|
sl@0
|
563 |
}
|
sl@0
|
564 |
if (aStream<*p)
|
sl@0
|
565 |
return;
|
sl@0
|
566 |
//
|
sl@0
|
567 |
for (;*p!=aStream;++p)
|
sl@0
|
568 |
{
|
sl@0
|
569 |
__ASSERT_DEBUG(p<iFinish,User::Invariant());
|
sl@0
|
570 |
__ASSERT_DEBUG(*p<aStream,User::Invariant());
|
sl@0
|
571 |
}
|
sl@0
|
572 |
if (p==iNext)
|
sl@0
|
573 |
iNext=p+1;
|
sl@0
|
574 |
else
|
sl@0
|
575 |
*p=-1;
|
sl@0
|
576 |
}
|
sl@0
|
577 |
|
sl@0
|
578 |
void TPermanentStoreStreamIter::Push(TInt* aHeap,TInt* aHole,TInt aValue)
|
sl@0
|
579 |
//
|
sl@0
|
580 |
// push aValue onto the heap (which will grow)
|
sl@0
|
581 |
//
|
sl@0
|
582 |
{
|
sl@0
|
583 |
TInt ix=aHole+1-aHeap;
|
sl@0
|
584 |
while (aHole>aHeap)
|
sl@0
|
585 |
{
|
sl@0
|
586 |
TInt link=ix-(ix>>1);
|
sl@0
|
587 |
ix-=link;
|
sl@0
|
588 |
TInt* parent=aHole-link;
|
sl@0
|
589 |
if (*parent>=aValue)
|
sl@0
|
590 |
break;
|
sl@0
|
591 |
*aHole=*parent;
|
sl@0
|
592 |
aHole=parent;
|
sl@0
|
593 |
}
|
sl@0
|
594 |
*aHole=aValue;
|
sl@0
|
595 |
}
|
sl@0
|
596 |
|
sl@0
|
597 |
TInt TPermanentStoreStreamIter::PopPush(TInt* aHeap,TInt* anEnd,TInt aValue)
|
sl@0
|
598 |
//
|
sl@0
|
599 |
// pop the top and push aValue
|
sl@0
|
600 |
//
|
sl@0
|
601 |
{
|
sl@0
|
602 |
TInt top=*aHeap;
|
sl@0
|
603 |
TInt* hole=aHeap;
|
sl@0
|
604 |
--anEnd;
|
sl@0
|
605 |
TInt ix=1;
|
sl@0
|
606 |
for (;;)
|
sl@0
|
607 |
{
|
sl@0
|
608 |
TInt* child=hole+ix;
|
sl@0
|
609 |
ix<<=1;
|
sl@0
|
610 |
if (child>anEnd)
|
sl@0
|
611 |
break;
|
sl@0
|
612 |
if (child<anEnd && *(child+1)>*child)
|
sl@0
|
613 |
{
|
sl@0
|
614 |
++ix;
|
sl@0
|
615 |
++child;
|
sl@0
|
616 |
}
|
sl@0
|
617 |
if (*child<=aValue)
|
sl@0
|
618 |
break;
|
sl@0
|
619 |
*hole=*child;
|
sl@0
|
620 |
hole=child;
|
sl@0
|
621 |
}
|
sl@0
|
622 |
*hole=aValue;
|
sl@0
|
623 |
return top;
|
sl@0
|
624 |
}
|
sl@0
|
625 |
|
sl@0
|
626 |
|
sl@0
|
627 |
|
sl@0
|
628 |
// Class CPermanentStoreCollector
|
sl@0
|
629 |
|
sl@0
|
630 |
CPermanentStoreCollector* CPermanentStoreCollector::CompactorL(CPermanentStoreCoord& aCoord)
|
sl@0
|
631 |
{
|
sl@0
|
632 |
CPermanentStoreCollector* self=ReclaimerL(aCoord);
|
sl@0
|
633 |
CleanupStack::PushL(self);
|
sl@0
|
634 |
self->iReloc=new(ELeave) TPermanentStoreRelocator;
|
sl@0
|
635 |
CleanupStack::Pop();
|
sl@0
|
636 |
return self;
|
sl@0
|
637 |
}
|
sl@0
|
638 |
|
sl@0
|
639 |
CPermanentStoreCollector* CPermanentStoreCollector::ReclaimerL(CPermanentStoreCoord& aCoord)
|
sl@0
|
640 |
{
|
sl@0
|
641 |
return new(ELeave) CPermanentStoreCollector(aCoord);
|
sl@0
|
642 |
}
|
sl@0
|
643 |
|
sl@0
|
644 |
CPermanentStoreCollector::CPermanentStoreCollector(CPermanentStoreCoord& aCoord)
|
sl@0
|
645 |
: iCoord(&aCoord),iHost(&aCoord.Host()),iStreams(EGranularity,_FOFF(TEntry,entry.ref))
|
sl@0
|
646 |
{
|
sl@0
|
647 |
Coord().Inc();
|
sl@0
|
648 |
}
|
sl@0
|
649 |
|
sl@0
|
650 |
CPermanentStoreCollector::~CPermanentStoreCollector()
|
sl@0
|
651 |
{
|
sl@0
|
652 |
delete iReloc;
|
sl@0
|
653 |
iStreams.Close();
|
sl@0
|
654 |
iMark.Withdraw(Host());
|
sl@0
|
655 |
Coord().Dec();
|
sl@0
|
656 |
}
|
sl@0
|
657 |
|
sl@0
|
658 |
void CPermanentStoreCollector::DoRelease()
|
sl@0
|
659 |
{
|
sl@0
|
660 |
delete this;
|
sl@0
|
661 |
}
|
sl@0
|
662 |
|
sl@0
|
663 |
void CPermanentStoreCollector::DoResetL(TInt& aCount)
|
sl@0
|
664 |
{
|
sl@0
|
665 |
iMark.Withdraw(Host());
|
sl@0
|
666 |
iMark=KStreamBeginning;
|
sl@0
|
667 |
iCoordGen=Coord().Generation();
|
sl@0
|
668 |
iFree=0;
|
sl@0
|
669 |
TRAPD(r, aCount = FastResetL());
|
sl@0
|
670 |
if (r == KErrNone)
|
sl@0
|
671 |
return;
|
sl@0
|
672 |
if (r != KErrNoMemory)
|
sl@0
|
673 |
User::Leave(r);
|
sl@0
|
674 |
// fall back to fixed memory solution
|
sl@0
|
675 |
iState=EGetFree;
|
sl@0
|
676 |
if (Compacting())
|
sl@0
|
677 |
iReloc->Reset();
|
sl@0
|
678 |
iIter.Reset();
|
sl@0
|
679 |
TInt streams=iIter.FillL(Coord().ConsolidateL());
|
sl@0
|
680 |
aCount=streams+1;
|
sl@0
|
681 |
}
|
sl@0
|
682 |
|
sl@0
|
683 |
const TInt KMaxStepEffort=9;
|
sl@0
|
684 |
|
sl@0
|
685 |
void CPermanentStoreCollector::DoNextL(TInt& aStep,TInt& aTotal)
|
sl@0
|
686 |
//
|
sl@0
|
687 |
// Dispatch the next set of operations
|
sl@0
|
688 |
//
|
sl@0
|
689 |
{
|
sl@0
|
690 |
if (aStep<=0)
|
sl@0
|
691 |
{
|
sl@0
|
692 |
__DEBUG(Panic(TStorePanic()));
|
sl@0
|
693 |
return;
|
sl@0
|
694 |
}
|
sl@0
|
695 |
//
|
sl@0
|
696 |
if (iCoordGen!=Coord().Generation() || Coord().TableL().IsVirtual())
|
sl@0
|
697 |
__LEAVE(KErrNotReady);
|
sl@0
|
698 |
//
|
sl@0
|
699 |
TInt effort=0;
|
sl@0
|
700 |
do
|
sl@0
|
701 |
{
|
sl@0
|
702 |
switch (iState)
|
sl@0
|
703 |
{
|
sl@0
|
704 |
default:
|
sl@0
|
705 |
__DEBUG(User::Invariant());
|
sl@0
|
706 |
case EGetFree:
|
sl@0
|
707 |
effort+=GetFreeL();
|
sl@0
|
708 |
break;
|
sl@0
|
709 |
case ESkip:
|
sl@0
|
710 |
effort+=SkipL(aTotal);
|
sl@0
|
711 |
--aStep;
|
sl@0
|
712 |
break;
|
sl@0
|
713 |
case EInitRelocator:
|
sl@0
|
714 |
effort+=InitRelocator();
|
sl@0
|
715 |
break;
|
sl@0
|
716 |
case EFillRelocator:
|
sl@0
|
717 |
effort+=FillRelocatorL();
|
sl@0
|
718 |
break;
|
sl@0
|
719 |
case EEvalRelocator:
|
sl@0
|
720 |
effort+=EvalRelocatorL();
|
sl@0
|
721 |
break;
|
sl@0
|
722 |
case EScanRelocator:
|
sl@0
|
723 |
effort+=ScanRelocator();
|
sl@0
|
724 |
break;
|
sl@0
|
725 |
case ERelocateStream:
|
sl@0
|
726 |
effort+=RelocateStreamL();
|
sl@0
|
727 |
--aStep;
|
sl@0
|
728 |
break;
|
sl@0
|
729 |
case ERelocateToc:
|
sl@0
|
730 |
RelocateTocL(aTotal);
|
sl@0
|
731 |
iStreams.Close();
|
sl@0
|
732 |
__ASSERT_DEBUG(aStep==1,User::Invariant());
|
sl@0
|
733 |
aStep=0;
|
sl@0
|
734 |
return;
|
sl@0
|
735 |
case EFastSort:
|
sl@0
|
736 |
FastSort();
|
sl@0
|
737 |
--aStep;
|
sl@0
|
738 |
return;
|
sl@0
|
739 |
case EFastExtent:
|
sl@0
|
740 |
FastExtentL(aTotal);
|
sl@0
|
741 |
--aStep;
|
sl@0
|
742 |
return;
|
sl@0
|
743 |
case EFastRelocate:
|
sl@0
|
744 |
FastRelocateL(aTotal);
|
sl@0
|
745 |
--aStep;
|
sl@0
|
746 |
return;
|
sl@0
|
747 |
}
|
sl@0
|
748 |
} while(effort<KMaxStepEffort);
|
sl@0
|
749 |
__ASSERT_DEBUG(aStep>=1,User::Invariant());
|
sl@0
|
750 |
}
|
sl@0
|
751 |
|
sl@0
|
752 |
TInt CPermanentStoreCollector::GetFreeL()
|
sl@0
|
753 |
//
|
sl@0
|
754 |
// find the end of the free space
|
sl@0
|
755 |
//
|
sl@0
|
756 |
{
|
sl@0
|
757 |
__ASSERT_DEBUG(iState==EGetFree,User::Invariant());
|
sl@0
|
758 |
__ASSERT_DEBUG(iFree>=0,User::Invariant());
|
sl@0
|
759 |
//
|
sl@0
|
760 |
TInt effort=0;
|
sl@0
|
761 |
iEnd=iIter.Next();
|
sl@0
|
762 |
if (iEnd<0)
|
sl@0
|
763 |
{
|
sl@0
|
764 |
if (iIter.FillL(Coord().ConsolidateL())==0)
|
sl@0
|
765 |
{
|
sl@0
|
766 |
iState=ERelocateToc; // no more streams
|
sl@0
|
767 |
return effort;
|
sl@0
|
768 |
}
|
sl@0
|
769 |
effort=KMaxStepEffort;
|
sl@0
|
770 |
iEnd=iIter.Next();
|
sl@0
|
771 |
__ASSERT_DEBUG(iEnd>=0,User::Invariant());
|
sl@0
|
772 |
}
|
sl@0
|
773 |
iState=Compacting() && HaveEnoughSpace() ? EInitRelocator : ESkip;
|
sl@0
|
774 |
return effort;
|
sl@0
|
775 |
}
|
sl@0
|
776 |
|
sl@0
|
777 |
TInt CPermanentStoreCollector::SkipL(TInt& aTotal)
|
sl@0
|
778 |
//
|
sl@0
|
779 |
// Find some free space, iEnd was the last stream extracted from concat
|
sl@0
|
780 |
//
|
sl@0
|
781 |
{
|
sl@0
|
782 |
__ASSERT_DEBUG(iState==ESkip,User::Invariant());
|
sl@0
|
783 |
__ASSERT_DEBUG(iFree>=0&&iFree<=iEnd,User::Invariant());
|
sl@0
|
784 |
//
|
sl@0
|
785 |
aTotal+=iEnd-iFree;
|
sl@0
|
786 |
iFree=SkipLink(ExtentL(iEnd));
|
sl@0
|
787 |
iState=EGetFree;
|
sl@0
|
788 |
return 1; // effort
|
sl@0
|
789 |
}
|
sl@0
|
790 |
|
sl@0
|
791 |
TInt CPermanentStoreCollector::InitRelocator()
|
sl@0
|
792 |
//
|
sl@0
|
793 |
// initialise the relocator for the free space
|
sl@0
|
794 |
//
|
sl@0
|
795 |
{
|
sl@0
|
796 |
__ASSERT_DEBUG(iState==EInitRelocator,User::Invariant());
|
sl@0
|
797 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
798 |
//
|
sl@0
|
799 |
iState=iReloc->Reset(iEnd) ? EScanRelocator : EFillRelocator;
|
sl@0
|
800 |
return 0; // no real work at all
|
sl@0
|
801 |
}
|
sl@0
|
802 |
|
sl@0
|
803 |
TInt CPermanentStoreCollector::FillRelocatorL()
|
sl@0
|
804 |
//
|
sl@0
|
805 |
// Fill the relocator table
|
sl@0
|
806 |
//
|
sl@0
|
807 |
{
|
sl@0
|
808 |
__ASSERT_DEBUG(iState==EFillRelocator,User::Invariant());
|
sl@0
|
809 |
__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
|
sl@0
|
810 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
811 |
//
|
sl@0
|
812 |
TBool more=iReloc->FillL(Coord().ConsolidateL());
|
sl@0
|
813 |
iState=more ? EEvalRelocator : ESkip;
|
sl@0
|
814 |
return more ? KMaxStepEffort : 0;
|
sl@0
|
815 |
}
|
sl@0
|
816 |
|
sl@0
|
817 |
TInt CPermanentStoreCollector::EvalRelocatorL()
|
sl@0
|
818 |
//
|
sl@0
|
819 |
// evaluate the extents for the relocator
|
sl@0
|
820 |
//
|
sl@0
|
821 |
{
|
sl@0
|
822 |
__ASSERT_DEBUG(iState==EEvalRelocator,User::Invariant());
|
sl@0
|
823 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
824 |
//
|
sl@0
|
825 |
iReloc->EvaluateLengthsL(Host(),iMark,Coord().Base());
|
sl@0
|
826 |
iState=EScanRelocator;
|
sl@0
|
827 |
return KMaxStepEffort;
|
sl@0
|
828 |
}
|
sl@0
|
829 |
|
sl@0
|
830 |
TInt CPermanentStoreCollector::ScanRelocator()
|
sl@0
|
831 |
//
|
sl@0
|
832 |
// find a stream that will fit
|
sl@0
|
833 |
//
|
sl@0
|
834 |
{
|
sl@0
|
835 |
__ASSERT_DEBUG(iState==EScanRelocator,User::Invariant());
|
sl@0
|
836 |
__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
|
sl@0
|
837 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
838 |
//
|
sl@0
|
839 |
iState=iReloc->FindFit(iFree,iEnd) ? ERelocateStream : EFillRelocator;
|
sl@0
|
840 |
return 0;
|
sl@0
|
841 |
}
|
sl@0
|
842 |
|
sl@0
|
843 |
TInt CPermanentStoreCollector::RelocateStreamL()
|
sl@0
|
844 |
//
|
sl@0
|
845 |
// find and relocate a stream
|
sl@0
|
846 |
//
|
sl@0
|
847 |
{
|
sl@0
|
848 |
__ASSERT_DEBUG(iState==ERelocateStream,User::Invariant());
|
sl@0
|
849 |
__ASSERT_DEBUG(iFree>=0&&iFree<iEnd,User::Invariant());
|
sl@0
|
850 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
851 |
//
|
sl@0
|
852 |
const TEntry& e = *iReloc->Current();
|
sl@0
|
853 |
RelocateStreamL(e, iEnd);
|
sl@0
|
854 |
iState=e.entry.ref==iEnd ? EGetFree : HaveEnoughSpace() ? EScanRelocator : ESkip;
|
sl@0
|
855 |
iIter.Relocated(e.entry.ref);
|
sl@0
|
856 |
iReloc->Relocated(&e);
|
sl@0
|
857 |
return KMaxStepEffort;
|
sl@0
|
858 |
}
|
sl@0
|
859 |
|
sl@0
|
860 |
void CPermanentStoreCollector::RelocateTocL(TInt& aTotal)
|
sl@0
|
861 |
//
|
sl@0
|
862 |
// relocate the toc - return any wasted bytes
|
sl@0
|
863 |
//
|
sl@0
|
864 |
{
|
sl@0
|
865 |
__ASSERT_DEBUG(iState==ERelocateToc,User::Invariant());
|
sl@0
|
866 |
__ASSERT_DEBUG(iFree>=0,User::Invariant());
|
sl@0
|
867 |
//
|
sl@0
|
868 |
TInt toc=Coord().iToc+KOffsetTocHeader;
|
sl@0
|
869 |
if (toc<0)
|
sl@0
|
870 |
return;
|
sl@0
|
871 |
//
|
sl@0
|
872 |
if (Compacting())
|
sl@0
|
873 |
{
|
sl@0
|
874 |
TInt len=Coord().TableL().Extent()-toc;
|
sl@0
|
875 |
if (Fits(iFree,toc,len))
|
sl@0
|
876 |
{
|
sl@0
|
877 |
RelocateL(toc, len, EFrameDescriptive16, toc);
|
sl@0
|
878 |
Coord().MoveL(iFree-KOffsetTocHeader,iFree + len);
|
sl@0
|
879 |
return;
|
sl@0
|
880 |
}
|
sl@0
|
881 |
}
|
sl@0
|
882 |
// don't move it
|
sl@0
|
883 |
aTotal += toc-iFree;
|
sl@0
|
884 |
}
|
sl@0
|
885 |
|
sl@0
|
886 |
TBool CPermanentStoreCollector::HaveEnoughSpace() const
|
sl@0
|
887 |
{
|
sl@0
|
888 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
889 |
//
|
sl@0
|
890 |
return SpaceFor(iFree,iEnd,iReloc->MinLength());
|
sl@0
|
891 |
}
|
sl@0
|
892 |
|
sl@0
|
893 |
TInt CPermanentStoreCollector::ExtentL(TInt aStream)
|
sl@0
|
894 |
//
|
sl@0
|
895 |
// Check if the relocator knows before scanning the file
|
sl@0
|
896 |
//
|
sl@0
|
897 |
{
|
sl@0
|
898 |
if (Compacting())
|
sl@0
|
899 |
{
|
sl@0
|
900 |
TInt ext=iReloc->Extent(aStream);
|
sl@0
|
901 |
if (ext>=0)
|
sl@0
|
902 |
return ext;
|
sl@0
|
903 |
}
|
sl@0
|
904 |
return ::ExtentL(Host(),iMark,Coord().Base(),aStream);
|
sl@0
|
905 |
}
|
sl@0
|
906 |
|
sl@0
|
907 |
/* relocate a stream into [iFree, aExtent)
|
sl@0
|
908 |
|
sl@0
|
909 |
During compaction, for each string which is to be moved from position A1 to B1, the sequence of operations is:
|
sl@0
|
910 |
|
sl@0
|
911 |
1. Copy stream S1 content from position A1 to position B1 . The copy never overlaps so the old stream content is still good at this point.
|
sl@0
|
912 |
2. Optionally rewrite the file header to state that stream S1 is being relocated to B1 (more about the ‘optional below’)
|
sl@0
|
913 |
3. Overwrite the TOC entry for S1 to state that the content is now at B1
|
sl@0
|
914 |
|
sl@0
|
915 |
This function completes 3 steps above and will be called again and again for every string to be moved.
|
sl@0
|
916 |
|
sl@0
|
917 |
In terms of data consistency, first consider the impact of a mid-write failure in any of these steps (when write caching is disabled):
|
sl@0
|
918 |
1. If step #1 only partially completes the file is good as the original content is intact and the new content was being written to otherwise free space
|
sl@0
|
919 |
2. If step #2 only partially completes the header CRC fails and only the TOC reference is considered valid (so the corrupt stream relocation record is ignored).
|
sl@0
|
920 |
The TOC will be good because it is being overwritten with the same content.
|
sl@0
|
921 |
3. If step #3 only partially completes the entry for S1 in the TOC is corrupt, BUT the relocation record for S1 in the file header is good and will
|
sl@0
|
922 |
override the entry in the TOC.
|
sl@0
|
923 |
|
sl@0
|
924 |
In all cases the file is never broken by a crash in mid-compaction.
|
sl@0
|
925 |
|
sl@0
|
926 |
Step #2 is optional – there are many cases when step #3 cannot fail ‘halfway through’ because the underlying media makes atomic block/page based
|
sl@0
|
927 |
updates and the write does not cross any block boundaries. In STORE we assume that blocks cannot be smaller than 512 bytes and any flash based
|
sl@0
|
928 |
media provides the required behavior. Thus 99% of the step #2 writes are eliminated.
|
sl@0
|
929 |
|
sl@0
|
930 |
Note that sequencing MATTERS even for just one stream. If the TOC update hits the disk before the content is moved, and then the device fails
|
sl@0
|
931 |
we will have a broken file: S1 points to B1 which contains garbage. Equally in the case where step #2 is required (i.e. when step #3 straddles
|
sl@0
|
932 |
a block boundary and could fail) step 2 has to go before the step 3. Otherwise write #3 could go to disk and fail part way through before write #2
|
sl@0
|
933 |
and leave the TOC corrupt with no recovery in the file header.
|
sl@0
|
934 |
|
sl@0
|
935 |
Consider the case that step 2 was omitted, so the Store relies on step 3 being completed in order to know that S1 is in location B1;
|
sl@0
|
936 |
and that no flush is done after step 3. In step 4 the stream S2 is moved – at this point the old space for stream S1 at A1 is considered empty
|
sl@0
|
937 |
– and suppose it gets moved from A2 to B2 where B2 overlaps/overwrites A1. If the writes in step 3 and step 4 are re-ordered and the step 3
|
sl@0
|
938 |
write does not happen – then the TOC will claim that S1 is still at A1 but this location in the file has been overwritten with data from S2.
|
sl@0
|
939 |
A corrupted file.
|
sl@0
|
940 |
|
sl@0
|
941 |
Based on the knowledge above, it is strongly recommended to set EFileWriteDirectIO bit when opening the file so that the order is maintained
|
sl@0
|
942 |
when writing to the file.
|
sl@0
|
943 |
*/
|
sl@0
|
944 |
void CPermanentStoreCollector::RelocateStreamL(const CPermanentStoreCollector::TEntry& aReloc, TInt aExtent)
|
sl@0
|
945 |
{
|
sl@0
|
946 |
if (Coord().Accessed()) // must have exclusive access to relocate the stream
|
sl@0
|
947 |
__LEAVE(KErrInUse);
|
sl@0
|
948 |
//
|
sl@0
|
949 |
TInt end=RelocateL(aReloc.entry.ref,aReloc.len,aReloc.entry.handle == KHandleTocBase ? EFrameDescriptive16 : EFrameData16, aExtent);
|
sl@0
|
950 |
//Step 1, 4,....
|
sl@0
|
951 |
Coord().RelocateL(aReloc.entry.handle, iFree);
|
sl@0
|
952 |
// Step 2 & 3, 5 & 6,...
|
sl@0
|
953 |
iCoordGen=Coord().Generation(); // changed by relocation
|
sl@0
|
954 |
iFree = end;
|
sl@0
|
955 |
}
|
sl@0
|
956 |
|
sl@0
|
957 |
TInt CPermanentStoreCollector::RelocateL(TInt aStream, TInt aLength, TFrameType16 aType, TInt aExtent)
|
sl@0
|
958 |
//
|
sl@0
|
959 |
// Relocate a data stream into [iFree, aExtent)
|
sl@0
|
960 |
//
|
sl@0
|
961 |
{
|
sl@0
|
962 |
__ASSERT_DEBUG(Fits(iFree,aExtent,aLength),User::Invariant());
|
sl@0
|
963 |
__ASSERT_DEBUG(Compacting(),User::Invariant());
|
sl@0
|
964 |
//
|
sl@0
|
965 |
TInt end=SkipLink(iFree+aLength);
|
sl@0
|
966 |
TInt terminator;
|
sl@0
|
967 |
if (end==aExtent)
|
sl@0
|
968 |
terminator=-1;
|
sl@0
|
969 |
else
|
sl@0
|
970 |
{
|
sl@0
|
971 |
TInt anchor=((end>>KShiftFrameLength16)+1)<<KShiftFrameLength16;
|
sl@0
|
972 |
if (aExtent<anchor)
|
sl@0
|
973 |
{
|
sl@0
|
974 |
__ASSERT_DEBUG(aExtent-KSizeFrameDes16-end>0,User::Invariant());
|
sl@0
|
975 |
terminator=EFrameFree16+(aExtent-KSizeFrameDes16-end);
|
sl@0
|
976 |
}
|
sl@0
|
977 |
else
|
sl@0
|
978 |
terminator=EFrameFree16+KFrameOpen16;
|
sl@0
|
979 |
}
|
sl@0
|
980 |
//
|
sl@0
|
981 |
RFrame16Buf from(Coord().Base());
|
sl@0
|
982 |
from.Set(Host(),aStream,aStream+aLength,MStreamBuf::ERead);
|
sl@0
|
983 |
from.PushL();
|
sl@0
|
984 |
TRelocatorInput input(Host(),iMark);
|
sl@0
|
985 |
input.OpenL(aType,Coord().Base(),iFree,aLength,terminator);
|
sl@0
|
986 |
from.ReadL(input,aLength);
|
sl@0
|
987 |
input.CommitL();
|
sl@0
|
988 |
CleanupStack::PopAndDestroy();
|
sl@0
|
989 |
//
|
sl@0
|
990 |
return end;
|
sl@0
|
991 |
}
|
sl@0
|
992 |
|
sl@0
|
993 |
|
sl@0
|
994 |
TInt CPermanentStoreCollector::FastResetL()
|
sl@0
|
995 |
//
|
sl@0
|
996 |
// Fill the table with the streams with data in the store
|
sl@0
|
997 |
//
|
sl@0
|
998 |
{
|
sl@0
|
999 |
iStreams.Reset();
|
sl@0
|
1000 |
//
|
sl@0
|
1001 |
CleanupClosePushL(iStreams);
|
sl@0
|
1002 |
RPermanentStoreTocIter iter(Coord().ConsolidateL());
|
sl@0
|
1003 |
CleanupReleasePushL(iter);
|
sl@0
|
1004 |
TEntry entry;
|
sl@0
|
1005 |
for (iter.ResetL();iter.NextL(entry.entry);)
|
sl@0
|
1006 |
{
|
sl@0
|
1007 |
if (entry.entry.handle<0)
|
sl@0
|
1008 |
continue;
|
sl@0
|
1009 |
if (entry.entry.ref<0)
|
sl@0
|
1010 |
continue;
|
sl@0
|
1011 |
User::LeaveIfError(iStreams.Append(entry));
|
sl@0
|
1012 |
}
|
sl@0
|
1013 |
CleanupStack::PopAndDestroy(&iter);
|
sl@0
|
1014 |
CleanupStack::Pop(&iStreams);
|
sl@0
|
1015 |
//
|
sl@0
|
1016 |
// always have final (toc) step
|
sl@0
|
1017 |
iState = ERelocateToc;
|
sl@0
|
1018 |
TInt streams = iStreams.Count();
|
sl@0
|
1019 |
if (streams == 0)
|
sl@0
|
1020 |
return 1;
|
sl@0
|
1021 |
iState = EFastSort;
|
sl@0
|
1022 |
// ordering is 1 step and evaluating the extents is several more
|
sl@0
|
1023 |
TInt count = 2 + (streams + EExtentStep - 1)/EExtentStep;
|
sl@0
|
1024 |
if (Compacting())
|
sl@0
|
1025 |
count += streams;
|
sl@0
|
1026 |
return count;
|
sl@0
|
1027 |
}
|
sl@0
|
1028 |
|
sl@0
|
1029 |
void CPermanentStoreCollector::FastSort()
|
sl@0
|
1030 |
{
|
sl@0
|
1031 |
__ASSERT_DEBUG(iState == EFastSort, User::Invariant());
|
sl@0
|
1032 |
//
|
sl@0
|
1033 |
iStreams.SortSigned();
|
sl@0
|
1034 |
iNext = &iStreams[0];
|
sl@0
|
1035 |
iLast = iNext + iStreams.Count();
|
sl@0
|
1036 |
iState = EFastExtent;
|
sl@0
|
1037 |
}
|
sl@0
|
1038 |
|
sl@0
|
1039 |
void CPermanentStoreCollector::FastExtentL(TInt& aTotal)
|
sl@0
|
1040 |
//
|
sl@0
|
1041 |
// Evaluate the lengths for all the streams
|
sl@0
|
1042 |
// if reclaiming, update aTotal with free space skipped
|
sl@0
|
1043 |
// return false until we've done the last one
|
sl@0
|
1044 |
//
|
sl@0
|
1045 |
{
|
sl@0
|
1046 |
__ASSERT_DEBUG(iState == EFastExtent, User::Invariant());
|
sl@0
|
1047 |
__ASSERT_DEBUG(iNext != iLast, User::Invariant());
|
sl@0
|
1048 |
|
sl@0
|
1049 |
TEntry* e = iNext;
|
sl@0
|
1050 |
const TEntry* end = Min(iLast, e + EExtentStep);
|
sl@0
|
1051 |
do
|
sl@0
|
1052 |
{
|
sl@0
|
1053 |
TInt ref = e->entry.ref;
|
sl@0
|
1054 |
__ASSERT_DEBUG(TUint(iFree)<=TUint(ref),User::Invariant());
|
sl@0
|
1055 |
TInt ext = ::ExtentL(Host(), iMark, Coord().Base(), ref);
|
sl@0
|
1056 |
e->len = ext - ref;
|
sl@0
|
1057 |
if (!Compacting())
|
sl@0
|
1058 |
aTotal += ref - iFree;
|
sl@0
|
1059 |
iFree = SkipLink(ext);
|
sl@0
|
1060 |
} while (++e < end);
|
sl@0
|
1061 |
iNext = e;
|
sl@0
|
1062 |
//
|
sl@0
|
1063 |
if (e == iLast)
|
sl@0
|
1064 |
{
|
sl@0
|
1065 |
if (!Compacting())
|
sl@0
|
1066 |
iState = ERelocateToc;
|
sl@0
|
1067 |
else
|
sl@0
|
1068 |
{
|
sl@0
|
1069 |
iNext = &iStreams[0];
|
sl@0
|
1070 |
iFree = 0;
|
sl@0
|
1071 |
iState = EFastRelocate;
|
sl@0
|
1072 |
}
|
sl@0
|
1073 |
}
|
sl@0
|
1074 |
}
|
sl@0
|
1075 |
|
sl@0
|
1076 |
CPermanentStoreCollector::TEntry* CPermanentStoreCollector::BestFit(TInt aPos, TInt aExt, TEntry* aFirst, TEntry* aLast)
|
sl@0
|
1077 |
//
|
sl@0
|
1078 |
// [aPos, aExt) is a hole - find the best fit to fill it in [aFirst, aLast)
|
sl@0
|
1079 |
//
|
sl@0
|
1080 |
{
|
sl@0
|
1081 |
__ASSERT_DEBUG(aPos <= aExt, User::Invariant());
|
sl@0
|
1082 |
TEntry* r = 0;
|
sl@0
|
1083 |
if (aPos == aExt)
|
sl@0
|
1084 |
return r;
|
sl@0
|
1085 |
//
|
sl@0
|
1086 |
if ((aExt & KMaskFrameLength16) != 0)
|
sl@0
|
1087 |
aExt -= KSizeFrameDes16;
|
sl@0
|
1088 |
const TInt mingap = Min(KSizeFrameDes16 + 1, (aExt & KMaskFrameLength16));
|
sl@0
|
1089 |
TInt rlen = 0;
|
sl@0
|
1090 |
TInt avail = aExt - aPos;
|
sl@0
|
1091 |
do
|
sl@0
|
1092 |
{
|
sl@0
|
1093 |
TInt len;
|
sl@0
|
1094 |
for (;;)
|
sl@0
|
1095 |
{
|
sl@0
|
1096 |
len = (--aLast)->len;
|
sl@0
|
1097 |
if (len > rlen)
|
sl@0
|
1098 |
break;
|
sl@0
|
1099 |
if (aFirst == aLast)
|
sl@0
|
1100 |
return r;
|
sl@0
|
1101 |
}
|
sl@0
|
1102 |
TInt d = avail - len;
|
sl@0
|
1103 |
if (d < 0)
|
sl@0
|
1104 |
continue;
|
sl@0
|
1105 |
if (d == 0) // exact fit
|
sl@0
|
1106 |
return aLast;
|
sl@0
|
1107 |
if (d < mingap)
|
sl@0
|
1108 |
continue;
|
sl@0
|
1109 |
r = aLast;
|
sl@0
|
1110 |
rlen = len;
|
sl@0
|
1111 |
} while (aFirst != aLast);
|
sl@0
|
1112 |
return r;
|
sl@0
|
1113 |
}
|
sl@0
|
1114 |
|
sl@0
|
1115 |
void CPermanentStoreCollector::FastRelocateL(TInt& aTotal)
|
sl@0
|
1116 |
//
|
sl@0
|
1117 |
// Look for a stream to move. Either move a stream or fail to find a fit before returning
|
sl@0
|
1118 |
// fill holes from the front of the file with the biggest block that will fit (inverted current algorithm)
|
sl@0
|
1119 |
// return true when the last hole has been looked at
|
sl@0
|
1120 |
//
|
sl@0
|
1121 |
{
|
sl@0
|
1122 |
__ASSERT_DEBUG(iState == EFastRelocate, User::Invariant());
|
sl@0
|
1123 |
__ASSERT_DEBUG(iNext != iLast, User::Invariant());
|
sl@0
|
1124 |
|
sl@0
|
1125 |
TEntry* e = iNext;
|
sl@0
|
1126 |
TInt ext = e->entry.ref;
|
sl@0
|
1127 |
__ASSERT_DEBUG(ext >= 0, User::Invariant());
|
sl@0
|
1128 |
|
sl@0
|
1129 |
TEntry* r = BestFit(iFree, ext, e, iLast);
|
sl@0
|
1130 |
if (!r)
|
sl@0
|
1131 |
{
|
sl@0
|
1132 |
// Nothing fits, accumulate free space
|
sl@0
|
1133 |
aTotal += ext - iFree;
|
sl@0
|
1134 |
iFree = SkipLink(ext + e->len);
|
sl@0
|
1135 |
}
|
sl@0
|
1136 |
else
|
sl@0
|
1137 |
{
|
sl@0
|
1138 |
// lets move it
|
sl@0
|
1139 |
RelocateStreamL(*r, ext);
|
sl@0
|
1140 |
if (r != e)
|
sl@0
|
1141 |
{
|
sl@0
|
1142 |
// relocated a stream other than the one terminating the current hole
|
sl@0
|
1143 |
// mark it at moved
|
sl@0
|
1144 |
r->entry.ref = -1;
|
sl@0
|
1145 |
r->len = -1;
|
sl@0
|
1146 |
return;
|
sl@0
|
1147 |
}
|
sl@0
|
1148 |
}
|
sl@0
|
1149 |
// skip through any relocated streams
|
sl@0
|
1150 |
while (++e < iLast && e->entry.ref < 0)
|
sl@0
|
1151 |
;
|
sl@0
|
1152 |
iNext = e;
|
sl@0
|
1153 |
if (e == iLast)
|
sl@0
|
1154 |
iState = ERelocateToc;
|
sl@0
|
1155 |
}
|