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 |
#include <s32mem.h>
|
sl@0
|
18 |
|
sl@0
|
19 |
#define UNUSED_VAR(a) a = a
|
sl@0
|
20 |
|
sl@0
|
21 |
const TInt KStreamDictGranularity=16;
|
sl@0
|
22 |
|
sl@0
|
23 |
NONSHARABLE_CLASS(HDictionaryStoreBuf) : public MStreamBuf
|
sl@0
|
24 |
{
|
sl@0
|
25 |
public:
|
sl@0
|
26 |
HDictionaryStoreBuf(MStreamBuf& aBuf,CDictionaryStore& aDic,TStreamId aStreamId,TUid aUid);
|
sl@0
|
27 |
private:
|
sl@0
|
28 |
// from MStreamBuf
|
sl@0
|
29 |
void DoSynchL(); // only update the stream dic during synch
|
sl@0
|
30 |
// just delegate the rest straight through...
|
sl@0
|
31 |
void DoRelease();
|
sl@0
|
32 |
TInt DoReadL(TAny* aPtr,TInt aMaxLength);
|
sl@0
|
33 |
TInt DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus);
|
sl@0
|
34 |
TStreamTransfer DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer);
|
sl@0
|
35 |
void DoWriteL(const TAny* aPtr,TInt aLength);
|
sl@0
|
36 |
TInt DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus);
|
sl@0
|
37 |
TStreamTransfer DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer);
|
sl@0
|
38 |
TStreamPos DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset);
|
sl@0
|
39 |
private:
|
sl@0
|
40 |
MStreamBuf* iRealBuf; // delegate everything to this
|
sl@0
|
41 |
CDictionaryStore* iDictionary;
|
sl@0
|
42 |
TStreamId iStreamId; // id of the actual stream in the store
|
sl@0
|
43 |
TUid iStreamUid; // Uid to be paired with the stream id in the dic
|
sl@0
|
44 |
};
|
sl@0
|
45 |
|
sl@0
|
46 |
NONSHARABLE_CLASS(HNullBuf) : public TMemBuf
|
sl@0
|
47 |
{
|
sl@0
|
48 |
public:
|
sl@0
|
49 |
HNullBuf();
|
sl@0
|
50 |
private:
|
sl@0
|
51 |
void DoRelease();
|
sl@0
|
52 |
};
|
sl@0
|
53 |
|
sl@0
|
54 |
////////////////////////////////////////
|
sl@0
|
55 |
// CStreamDictionary
|
sl@0
|
56 |
////////////////////////////////////////
|
sl@0
|
57 |
|
sl@0
|
58 |
EXPORT_C CStreamDictionary* CStreamDictionary::NewL()
|
sl@0
|
59 |
/** Allocates, constructs and returns a pointer to a new stream dictionary.
|
sl@0
|
60 |
|
sl@0
|
61 |
The function leaves if it cannot complete successfully.
|
sl@0
|
62 |
|
sl@0
|
63 |
@return A pointer to the new stream dictionary object. */
|
sl@0
|
64 |
{
|
sl@0
|
65 |
return new(ELeave) CStreamDictionary;
|
sl@0
|
66 |
}
|
sl@0
|
67 |
|
sl@0
|
68 |
EXPORT_C CStreamDictionary* CStreamDictionary::NewLC()
|
sl@0
|
69 |
/** Allocates, constructs and returns a pointer to a new stream dictionary, putting
|
sl@0
|
70 |
the pointer onto the cleanup stack.
|
sl@0
|
71 |
|
sl@0
|
72 |
Putting the pointer onto the cleanup stack allows the object and allocated
|
sl@0
|
73 |
resources to be cleaned up if a subsequent leave occurs.
|
sl@0
|
74 |
|
sl@0
|
75 |
The function leaves if it cannot complete successfully.
|
sl@0
|
76 |
|
sl@0
|
77 |
@return A pointer to the new stream dictionary object. */
|
sl@0
|
78 |
{
|
sl@0
|
79 |
CStreamDictionary* dict=NewL();
|
sl@0
|
80 |
CleanupStack::PushL(dict);
|
sl@0
|
81 |
return dict;
|
sl@0
|
82 |
}
|
sl@0
|
83 |
|
sl@0
|
84 |
EXPORT_C CStreamDictionary::CStreamDictionary()
|
sl@0
|
85 |
: iArray(KStreamDictGranularity)
|
sl@0
|
86 |
{}
|
sl@0
|
87 |
|
sl@0
|
88 |
EXPORT_C CStreamDictionary::~CStreamDictionary()
|
sl@0
|
89 |
/** Frees resources owned by the object, prior to its destruction. */
|
sl@0
|
90 |
{}
|
sl@0
|
91 |
|
sl@0
|
92 |
EXPORT_C void CStreamDictionary::AssignL(TUid aUid,TStreamId anId)
|
sl@0
|
93 |
/** Creates or changes an association between a UID and a stream id.
|
sl@0
|
94 |
|
sl@0
|
95 |
If the stream dictionary already contains an entry with the same UID as aUid,
|
sl@0
|
96 |
then the associated stream id in that entry is replaced by anId.
|
sl@0
|
97 |
|
sl@0
|
98 |
If anId has the value KNullStreamId, then the function attempts to remove
|
sl@0
|
99 |
the entry whose unique identifier matches aUid.
|
sl@0
|
100 |
|
sl@0
|
101 |
@param aUid The UID
|
sl@0
|
102 |
@param anId The stream id.
|
sl@0
|
103 |
@see CStreamDictionary::Remove() */
|
sl@0
|
104 |
{
|
sl@0
|
105 |
if (anId==KNullStreamId)
|
sl@0
|
106 |
{
|
sl@0
|
107 |
Remove(aUid); // default associated stream is null
|
sl@0
|
108 |
return;
|
sl@0
|
109 |
}
|
sl@0
|
110 |
//
|
sl@0
|
111 |
TEntry entry;
|
sl@0
|
112 |
entry.iUid=aUid;
|
sl@0
|
113 |
TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
|
sl@0
|
114 |
TInt i;
|
sl@0
|
115 |
if (iArray.Find(entry,key,i)==0)
|
sl@0
|
116 |
{
|
sl@0
|
117 |
iArray[i].iId=anId;
|
sl@0
|
118 |
return;
|
sl@0
|
119 |
}
|
sl@0
|
120 |
//
|
sl@0
|
121 |
__ASSERT_DEBUG(i==iArray.Count(),User::Invariant());
|
sl@0
|
122 |
entry.iId=anId;
|
sl@0
|
123 |
iArray.InsertL(i,entry);
|
sl@0
|
124 |
}
|
sl@0
|
125 |
|
sl@0
|
126 |
EXPORT_C void CStreamDictionary::Remove(TUid aUid)
|
sl@0
|
127 |
/** Removes an association from the stream dictionary.
|
sl@0
|
128 |
|
sl@0
|
129 |
The function searches the stream dictionary for an entry whose UID matches
|
sl@0
|
130 |
aUid. If a match is found, the entry is removed. If no match is found, the
|
sl@0
|
131 |
stream dictionary remains unchanged.
|
sl@0
|
132 |
|
sl@0
|
133 |
@param aUid The UID whose matching entry is to be removed from the stream
|
sl@0
|
134 |
dictionary. */
|
sl@0
|
135 |
{
|
sl@0
|
136 |
TEntry entry;
|
sl@0
|
137 |
entry.iUid=aUid;
|
sl@0
|
138 |
TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
|
sl@0
|
139 |
TInt i;
|
sl@0
|
140 |
if (iArray.Find(entry,key,i)==0)
|
sl@0
|
141 |
iArray.Delete(i);
|
sl@0
|
142 |
}
|
sl@0
|
143 |
|
sl@0
|
144 |
EXPORT_C TStreamId CStreamDictionary::At(TUid aUid) const
|
sl@0
|
145 |
/** Returns the stream id associated with the specified UID.
|
sl@0
|
146 |
|
sl@0
|
147 |
@param aUid The UID whose associated stream id is to be returned.
|
sl@0
|
148 |
@return The stream id associated with the specified UID.KNullStreamId, if no
|
sl@0
|
149 |
entry in the stream dictionary contains the specified UID. */
|
sl@0
|
150 |
{
|
sl@0
|
151 |
TEntry entry;
|
sl@0
|
152 |
entry.iUid=aUid;
|
sl@0
|
153 |
TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
|
sl@0
|
154 |
TInt i;
|
sl@0
|
155 |
if (iArray.Find(entry,key,i)!=0)
|
sl@0
|
156 |
return KNullStreamId;
|
sl@0
|
157 |
//
|
sl@0
|
158 |
return iArray[i].iId;
|
sl@0
|
159 |
}
|
sl@0
|
160 |
|
sl@0
|
161 |
EXPORT_C TBool CStreamDictionary::IsNull() const
|
sl@0
|
162 |
/** Tests whether the stream dictionary is empty.
|
sl@0
|
163 |
|
sl@0
|
164 |
@return True, if the stream dictionary is empty. False if the stream dictionary
|
sl@0
|
165 |
contains at least one entry. */
|
sl@0
|
166 |
{
|
sl@0
|
167 |
return iArray.Count()==0;
|
sl@0
|
168 |
}
|
sl@0
|
169 |
|
sl@0
|
170 |
EXPORT_C void CStreamDictionary::ExternalizeL(RWriteStream& aStream) const
|
sl@0
|
171 |
/** Externalises an object of this class to a write stream.
|
sl@0
|
172 |
|
sl@0
|
173 |
The presence of this function means that the standard templated operator<<()
|
sl@0
|
174 |
can be used to externalise objects of this class.
|
sl@0
|
175 |
|
sl@0
|
176 |
@param aStream Stream to which the object should be externalised */
|
sl@0
|
177 |
{
|
sl@0
|
178 |
aStream<<iArray;
|
sl@0
|
179 |
}
|
sl@0
|
180 |
|
sl@0
|
181 |
EXPORT_C void CStreamDictionary::InternalizeL(RReadStream& aStream)
|
sl@0
|
182 |
/** Internalises an object of this class from a read stream.
|
sl@0
|
183 |
|
sl@0
|
184 |
The presence of this function means that the standard templated operator>>()
|
sl@0
|
185 |
can be used to internalise objects of this class.
|
sl@0
|
186 |
|
sl@0
|
187 |
Note that this function has assignment semantics. It replaces the old value
|
sl@0
|
188 |
of the object with a new value read from the read stream.
|
sl@0
|
189 |
|
sl@0
|
190 |
@param aStream Stream from which the object should be internalised. */
|
sl@0
|
191 |
{
|
sl@0
|
192 |
aStream>>iArray;
|
sl@0
|
193 |
}
|
sl@0
|
194 |
|
sl@0
|
195 |
void CStreamDictionary::TEntry::ExternalizeL(RWriteStream& aStream) const
|
sl@0
|
196 |
{
|
sl@0
|
197 |
aStream<<iUid;
|
sl@0
|
198 |
aStream<<iId;
|
sl@0
|
199 |
}
|
sl@0
|
200 |
|
sl@0
|
201 |
void CStreamDictionary::TEntry::InternalizeL(RReadStream& aStream)
|
sl@0
|
202 |
{
|
sl@0
|
203 |
aStream>>iUid;
|
sl@0
|
204 |
aStream>>iId;
|
sl@0
|
205 |
}
|
sl@0
|
206 |
|
sl@0
|
207 |
|
sl@0
|
208 |
////////////////////////////////////////
|
sl@0
|
209 |
// CDictionaryStore
|
sl@0
|
210 |
////////////////////////////////////////
|
sl@0
|
211 |
|
sl@0
|
212 |
EXPORT_C CDictionaryStore::~CDictionaryStore()
|
sl@0
|
213 |
/** Frees resources owned by the object, prior to its destruction. */
|
sl@0
|
214 |
{
|
sl@0
|
215 |
delete iDictionary;
|
sl@0
|
216 |
delete iStore;
|
sl@0
|
217 |
}
|
sl@0
|
218 |
|
sl@0
|
219 |
|
sl@0
|
220 |
EXPORT_C void CDictionaryStore::ConstructL()
|
sl@0
|
221 |
{
|
sl@0
|
222 |
iDictionary = CStreamDictionary::NewL();
|
sl@0
|
223 |
RStoreWriteStream stream;
|
sl@0
|
224 |
iStore->SetRootL(stream.CreateLC(*iStore));
|
sl@0
|
225 |
stream<<*iDictionary;
|
sl@0
|
226 |
stream.CommitL();
|
sl@0
|
227 |
CleanupStack::PopAndDestroy(); // dictionary stream
|
sl@0
|
228 |
iStore->CommitL();
|
sl@0
|
229 |
}
|
sl@0
|
230 |
|
sl@0
|
231 |
|
sl@0
|
232 |
EXPORT_C void CDictionaryStore::Remove(TUid aUid)
|
sl@0
|
233 |
/** Removes the stream associated with the specified UID from the dictionary store.
|
sl@0
|
234 |
|
sl@0
|
235 |
If there is no stream associated with the specified UID, then the dictionary
|
sl@0
|
236 |
store remains unchanged.
|
sl@0
|
237 |
|
sl@0
|
238 |
The function cannot leave; it returns whether or not it is succesful.
|
sl@0
|
239 |
|
sl@0
|
240 |
@param aUid The UID whose associated stream is to be removed from the dictionary
|
sl@0
|
241 |
store. */
|
sl@0
|
242 |
{
|
sl@0
|
243 |
TRAPD(ignore,RemoveL(aUid));
|
sl@0
|
244 |
UNUSED_VAR(ignore);
|
sl@0
|
245 |
}
|
sl@0
|
246 |
|
sl@0
|
247 |
|
sl@0
|
248 |
EXPORT_C void CDictionaryStore::RemoveL(TUid aUid)
|
sl@0
|
249 |
/** Removes the stream associated with the specified UID from the dictionary store
|
sl@0
|
250 |
and leaves if unsuccessful.
|
sl@0
|
251 |
|
sl@0
|
252 |
If there is no stream associated with the specified UID, then the dictionary
|
sl@0
|
253 |
store remains unchanged.
|
sl@0
|
254 |
|
sl@0
|
255 |
@param aUid The UID whose associated stream is to be removed from the dictionary
|
sl@0
|
256 |
store. */
|
sl@0
|
257 |
{
|
sl@0
|
258 |
TStreamId id = DictionaryL()->At(aUid);
|
sl@0
|
259 |
if (id!=KNullStreamId)
|
sl@0
|
260 |
{
|
sl@0
|
261 |
DictionaryL()->Remove(aUid);
|
sl@0
|
262 |
iStore->DeleteL(id);
|
sl@0
|
263 |
iDictionaryHasChanged=ETrue;
|
sl@0
|
264 |
}
|
sl@0
|
265 |
}
|
sl@0
|
266 |
|
sl@0
|
267 |
|
sl@0
|
268 |
EXPORT_C TInt CDictionaryStore::Commit()
|
sl@0
|
269 |
/** Commits changes.
|
sl@0
|
270 |
|
sl@0
|
271 |
It establishes a new commit point and then compacts the dictionary store.
|
sl@0
|
272 |
Typically, this is done after changes to new or existing streams are complete
|
sl@0
|
273 |
and the streams themselves have been committed.
|
sl@0
|
274 |
|
sl@0
|
275 |
Establishing a new commit point makes changes to the store permanent. Until
|
sl@0
|
276 |
such changes are committed, they can be rolled back, effectively causing the
|
sl@0
|
277 |
store to revert back to its state before the changes were made.
|
sl@0
|
278 |
|
sl@0
|
279 |
This ensures that persistent data moves from one consistent state to another
|
sl@0
|
280 |
and guarantees the integrity of persistent store data in the event of failures.
|
sl@0
|
281 |
In particular, if a process terminates or a media failure occurs, the store
|
sl@0
|
282 |
reverts automatically to its state at the last successful commit point.
|
sl@0
|
283 |
|
sl@0
|
284 |
@return KErrNone if successful, otherwise another of the system-wide error
|
sl@0
|
285 |
codes. */
|
sl@0
|
286 |
{
|
sl@0
|
287 |
TRAPD(ret,CommitL());
|
sl@0
|
288 |
return ret;
|
sl@0
|
289 |
}
|
sl@0
|
290 |
|
sl@0
|
291 |
|
sl@0
|
292 |
EXPORT_C void CDictionaryStore::CommitL()
|
sl@0
|
293 |
/** Commits changes and leaves if unsuccessful.
|
sl@0
|
294 |
|
sl@0
|
295 |
@see CDictionaryStore::Commit() */
|
sl@0
|
296 |
{
|
sl@0
|
297 |
if (iDictionaryHasChanged)
|
sl@0
|
298 |
{
|
sl@0
|
299 |
// rewrite the root stream
|
sl@0
|
300 |
RStoreWriteStream stream;
|
sl@0
|
301 |
stream.ReplaceLC(*iStore,iStore->Root());
|
sl@0
|
302 |
stream<< *DictionaryL();
|
sl@0
|
303 |
stream.CommitL();
|
sl@0
|
304 |
CleanupStack::PopAndDestroy(); // dictionary stream
|
sl@0
|
305 |
}
|
sl@0
|
306 |
// commit the store
|
sl@0
|
307 |
iStore->CommitL();
|
sl@0
|
308 |
// compact the store, ignoring any failure (commit was good)
|
sl@0
|
309 |
TRAPD(
|
sl@0
|
310 |
ignore,
|
sl@0
|
311 |
{
|
sl@0
|
312 |
if (iStore->ReclaimL() >= KDictionaryCommitThreshold)
|
sl@0
|
313 |
{
|
sl@0
|
314 |
iStore->CompactL();
|
sl@0
|
315 |
}
|
sl@0
|
316 |
iStore->CommitL();
|
sl@0
|
317 |
} );
|
sl@0
|
318 |
UNUSED_VAR(ignore);
|
sl@0
|
319 |
}
|
sl@0
|
320 |
|
sl@0
|
321 |
|
sl@0
|
322 |
EXPORT_C void CDictionaryStore::Revert()
|
sl@0
|
323 |
/** Rolls the dictionary store back to its state at the last commit point.
|
sl@0
|
324 |
|
sl@0
|
325 |
A commit point is set using the Commit() or CommitL() functions.
|
sl@0
|
326 |
|
sl@0
|
327 |
The function returns, whether or not it completes successfully.
|
sl@0
|
328 |
|
sl@0
|
329 |
@see CDictionaryStore::Commit()
|
sl@0
|
330 |
@see CDictionaryStore::CommitL() */
|
sl@0
|
331 |
{
|
sl@0
|
332 |
TRAPD(ignore,RevertL());
|
sl@0
|
333 |
UNUSED_VAR(ignore);
|
sl@0
|
334 |
}
|
sl@0
|
335 |
|
sl@0
|
336 |
|
sl@0
|
337 |
EXPORT_C void CDictionaryStore::RevertL()
|
sl@0
|
338 |
/** Rolls the dictionary store back to its state at the last commit point and leaves
|
sl@0
|
339 |
if unsuccessful.
|
sl@0
|
340 |
|
sl@0
|
341 |
A commit point is set using the Commit() or CommitL() functions.
|
sl@0
|
342 |
|
sl@0
|
343 |
@see CDictionaryStore::Commit()
|
sl@0
|
344 |
@see CDictionaryStore::CommitL() */
|
sl@0
|
345 |
{
|
sl@0
|
346 |
delete iDictionary;
|
sl@0
|
347 |
iDictionary = NULL;
|
sl@0
|
348 |
iDictionaryHasChanged = EFalse;
|
sl@0
|
349 |
iStore->RevertL();
|
sl@0
|
350 |
}
|
sl@0
|
351 |
|
sl@0
|
352 |
|
sl@0
|
353 |
CStreamDictionary* CDictionaryStore::DictionaryL() const
|
sl@0
|
354 |
// Returns the handle of the stream dictionary, internalizing it first if necessary
|
sl@0
|
355 |
{
|
sl@0
|
356 |
if (!iDictionary)
|
sl@0
|
357 |
{
|
sl@0
|
358 |
// create a temporary dictionary
|
sl@0
|
359 |
CStreamDictionary* dictionary = CStreamDictionary::NewLC();
|
sl@0
|
360 |
// stream into the dictionary
|
sl@0
|
361 |
RStoreReadStream root;
|
sl@0
|
362 |
root.OpenLC(*iStore,iStore->Root());
|
sl@0
|
363 |
root>> *dictionary;
|
sl@0
|
364 |
root.Close();
|
sl@0
|
365 |
CleanupStack::PopAndDestroy(); // root
|
sl@0
|
366 |
// set iDictionary
|
sl@0
|
367 |
MUTABLE_CAST(CStreamDictionary*&,iDictionary) = dictionary;
|
sl@0
|
368 |
CleanupStack::Pop(); // dictionary
|
sl@0
|
369 |
}
|
sl@0
|
370 |
return iDictionary;
|
sl@0
|
371 |
}
|
sl@0
|
372 |
|
sl@0
|
373 |
|
sl@0
|
374 |
EXPORT_C TBool CDictionaryStore::IsNullL() const
|
sl@0
|
375 |
/** Tests whether the dictionary stores stream dictionary is empty.
|
sl@0
|
376 |
|
sl@0
|
377 |
@return True, if the stream dictionary is empty. False, if the stream dictionary
|
sl@0
|
378 |
contains at least one entry.
|
sl@0
|
379 |
@see CStreamDictionary */
|
sl@0
|
380 |
{
|
sl@0
|
381 |
return DictionaryL()->IsNull();
|
sl@0
|
382 |
}
|
sl@0
|
383 |
|
sl@0
|
384 |
|
sl@0
|
385 |
EXPORT_C TBool CDictionaryStore::IsPresentL(TUid aUid) const
|
sl@0
|
386 |
/** Tests whether the specified UID has an associated stream within this dictionary
|
sl@0
|
387 |
store.
|
sl@0
|
388 |
|
sl@0
|
389 |
@param aUid The UID.
|
sl@0
|
390 |
@return True, if there is a stream associated with the specified UID in the
|
sl@0
|
391 |
dictionary store; false otherwise.
|
sl@0
|
392 |
@see CStreamDictionary */
|
sl@0
|
393 |
{
|
sl@0
|
394 |
return DictionaryL()->At(aUid)!=KNullStreamId;
|
sl@0
|
395 |
}
|
sl@0
|
396 |
|
sl@0
|
397 |
|
sl@0
|
398 |
MStreamBuf* CDictionaryStore::GetSourceL(TUid aUid) const
|
sl@0
|
399 |
// Opens the stream matching aUid and returns its source.
|
sl@0
|
400 |
// If no stream matches aUid an empty stream buf is returned
|
sl@0
|
401 |
// Ownership of the source is transfered to the caller.
|
sl@0
|
402 |
{
|
sl@0
|
403 |
// get the id of the stream
|
sl@0
|
404 |
TStreamId id = DictionaryL()->At(aUid);
|
sl@0
|
405 |
// open the stream if it exists, otherwise open a dummy empty stream
|
sl@0
|
406 |
if (id==KNullStreamId)
|
sl@0
|
407 |
return (new(ELeave) HNullBuf);
|
sl@0
|
408 |
else
|
sl@0
|
409 |
{
|
sl@0
|
410 |
// open the stream and steal its source
|
sl@0
|
411 |
RStoreReadStream stream;
|
sl@0
|
412 |
stream.OpenL(*iStore,id);
|
sl@0
|
413 |
return stream.Source();
|
sl@0
|
414 |
}
|
sl@0
|
415 |
}
|
sl@0
|
416 |
|
sl@0
|
417 |
|
sl@0
|
418 |
MStreamBuf* CDictionaryStore::GetSinkL(TUid aUid)
|
sl@0
|
419 |
// If a stream matches aUid it is replaced ad its sink returned
|
sl@0
|
420 |
// If no stream matches aUid a new stream is created and its sink returned
|
sl@0
|
421 |
// Ownership of the sink is transfered to the caller.
|
sl@0
|
422 |
{
|
sl@0
|
423 |
// get the id of the stream
|
sl@0
|
424 |
TStreamId id = DictionaryL()->At(aUid);
|
sl@0
|
425 |
RStoreWriteStream stream;
|
sl@0
|
426 |
if (id!=KNullStreamId)
|
sl@0
|
427 |
//replace the stream
|
sl@0
|
428 |
stream.ReplaceLC(*iStore,id);
|
sl@0
|
429 |
else
|
sl@0
|
430 |
// create a new stream
|
sl@0
|
431 |
id = stream.CreateLC(*iStore);
|
sl@0
|
432 |
// create a dictionary store buffer to hold the incoming data
|
sl@0
|
433 |
MStreamBuf* buf = new(ELeave) HDictionaryStoreBuf(*stream.Sink(),*this,id,aUid);
|
sl@0
|
434 |
CleanupStack::Pop(); // stream;
|
sl@0
|
435 |
return buf;
|
sl@0
|
436 |
}
|
sl@0
|
437 |
|
sl@0
|
438 |
|
sl@0
|
439 |
//////////////////////////////////////////////
|
sl@0
|
440 |
// RDictionaryReadStream
|
sl@0
|
441 |
//////////////////////////////////////////////
|
sl@0
|
442 |
|
sl@0
|
443 |
EXPORT_C void RDictionaryReadStream::OpenL(const CDictionaryStore& aDictStore,TUid aUid)
|
sl@0
|
444 |
// Open the stream if it exists, otherwise open a dummy empty stream
|
sl@0
|
445 |
/** Opens the stream associated with the specified UID in the specified dictionary
|
sl@0
|
446 |
store, and prepares it for reading.
|
sl@0
|
447 |
|
sl@0
|
448 |
@param aDictStore The dictionary store containing the stream.
|
sl@0
|
449 |
@param aUid The unique identifier associated with the stream to be read. */
|
sl@0
|
450 |
{
|
sl@0
|
451 |
Attach(aDictStore.GetSourceL(aUid));
|
sl@0
|
452 |
}
|
sl@0
|
453 |
|
sl@0
|
454 |
|
sl@0
|
455 |
EXPORT_C void RDictionaryReadStream::OpenLC(const CDictionaryStore& aDictStore,TUid aUid)
|
sl@0
|
456 |
/** Opens the stream associated with the specified UID in the specified dictionary
|
sl@0
|
457 |
store, prepares it for reading, and puts a a cleanup item onto the cleanup
|
sl@0
|
458 |
stack..
|
sl@0
|
459 |
|
sl@0
|
460 |
Placing a cleanup item on the cleanup stack allows allocated resources to
|
sl@0
|
461 |
be cleaned up if a subsequent leave occurs.
|
sl@0
|
462 |
|
sl@0
|
463 |
@param aDictStore The dictionary store containing the stream.
|
sl@0
|
464 |
@param aUid The unique identifier associated with the stream to be read. */
|
sl@0
|
465 |
{
|
sl@0
|
466 |
OpenL(aDictStore,aUid);
|
sl@0
|
467 |
PushL();
|
sl@0
|
468 |
}
|
sl@0
|
469 |
|
sl@0
|
470 |
|
sl@0
|
471 |
//////////////////////////////////////////////
|
sl@0
|
472 |
// RDictionaryWriteStream
|
sl@0
|
473 |
//////////////////////////////////////////////
|
sl@0
|
474 |
|
sl@0
|
475 |
EXPORT_C void RDictionaryWriteStream::AssignL(CDictionaryStore& aDictStore,TUid aUid)
|
sl@0
|
476 |
// Replace the stream if it exists, otherwise create a new one
|
sl@0
|
477 |
/** Prepares a stream in the specified dictionary store for writing.
|
sl@0
|
478 |
|
sl@0
|
479 |
If no stream is associated with the specified UID, then a new stream is created
|
sl@0
|
480 |
and an association is made between the resulting stream id and the specified
|
sl@0
|
481 |
UID. If a stream is currently associated with the specified UID, then this
|
sl@0
|
482 |
existing stream is prepared for replacement.
|
sl@0
|
483 |
|
sl@0
|
484 |
@param aDictStore The dictionary store which contains the new or replacement
|
sl@0
|
485 |
stream.
|
sl@0
|
486 |
@param aUid The UID associated with the stream.
|
sl@0
|
487 |
@see RWriteStream::Release() */
|
sl@0
|
488 |
{
|
sl@0
|
489 |
Attach(aDictStore.GetSinkL(aUid));
|
sl@0
|
490 |
}
|
sl@0
|
491 |
|
sl@0
|
492 |
|
sl@0
|
493 |
EXPORT_C void RDictionaryWriteStream::AssignLC(CDictionaryStore& aDictStore,TUid aUid)
|
sl@0
|
494 |
/** Prepares a stream in the specified dictionary store for writing, and places
|
sl@0
|
495 |
a cleanup item for this RDictionaryWriteStream object onto the cleanup stack.
|
sl@0
|
496 |
|
sl@0
|
497 |
If no stream is associated with the specified UID, then a new stream is created
|
sl@0
|
498 |
and an association is made between the resulting stream id and the specified
|
sl@0
|
499 |
UID. If a stream is currently associated with the specified UID, then this
|
sl@0
|
500 |
existing stream is prepared for replacement.
|
sl@0
|
501 |
|
sl@0
|
502 |
Placing a cleanup item onto the cleanup stack allows allocated resources to
|
sl@0
|
503 |
be cleaned up if a subsequent leave occurs.
|
sl@0
|
504 |
|
sl@0
|
505 |
@param aDictStore The dictionary store which contains the new or replacement
|
sl@0
|
506 |
stream.
|
sl@0
|
507 |
@param aUid The UID associated with the stream. */
|
sl@0
|
508 |
{
|
sl@0
|
509 |
AssignL(aDictStore,aUid);
|
sl@0
|
510 |
PushL();
|
sl@0
|
511 |
}
|
sl@0
|
512 |
|
sl@0
|
513 |
|
sl@0
|
514 |
//////////////////////////////////////////////
|
sl@0
|
515 |
// HDictionaryStoreBuf
|
sl@0
|
516 |
//////////////////////////////////////////////
|
sl@0
|
517 |
|
sl@0
|
518 |
HDictionaryStoreBuf::HDictionaryStoreBuf(MStreamBuf& aBuf,CDictionaryStore& aDic,TStreamId aStreamId,TUid aUid)
|
sl@0
|
519 |
:iRealBuf(&aBuf),
|
sl@0
|
520 |
iDictionary(&aDic),
|
sl@0
|
521 |
iStreamId(aStreamId),
|
sl@0
|
522 |
iStreamUid(aUid)
|
sl@0
|
523 |
{}
|
sl@0
|
524 |
|
sl@0
|
525 |
|
sl@0
|
526 |
void HDictionaryStoreBuf::DoSynchL()
|
sl@0
|
527 |
// only update the stream dic during synch - this is when the stream gets added to the store
|
sl@0
|
528 |
{
|
sl@0
|
529 |
CStreamDictionary& dic = *iDictionary->DictionaryL();
|
sl@0
|
530 |
TBool newEntry = (dic.At(iStreamUid)==KNullStreamId);
|
sl@0
|
531 |
if (iRealBuf->SizeL()>0)
|
sl@0
|
532 |
{
|
sl@0
|
533 |
dic.AssignL(iStreamUid,iStreamId);
|
sl@0
|
534 |
TInt ret = iRealBuf->Synch();
|
sl@0
|
535 |
if (ret!=KErrNone)
|
sl@0
|
536 |
{// attempt rollback
|
sl@0
|
537 |
if (newEntry)
|
sl@0
|
538 |
dic.Remove(iStreamUid);
|
sl@0
|
539 |
__LEAVE(ret);
|
sl@0
|
540 |
}
|
sl@0
|
541 |
if (newEntry)
|
sl@0
|
542 |
iDictionary->iDictionaryHasChanged=ETrue;
|
sl@0
|
543 |
}
|
sl@0
|
544 |
else
|
sl@0
|
545 |
{
|
sl@0
|
546 |
iRealBuf->SynchL();
|
sl@0
|
547 |
iDictionary->iStore->Delete(iStreamId);
|
sl@0
|
548 |
iDictionary->Remove(iStreamUid);
|
sl@0
|
549 |
}
|
sl@0
|
550 |
}
|
sl@0
|
551 |
|
sl@0
|
552 |
|
sl@0
|
553 |
void HDictionaryStoreBuf::DoRelease()
|
sl@0
|
554 |
{
|
sl@0
|
555 |
iRealBuf->Release();
|
sl@0
|
556 |
delete this;
|
sl@0
|
557 |
}
|
sl@0
|
558 |
|
sl@0
|
559 |
|
sl@0
|
560 |
TInt HDictionaryStoreBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
|
sl@0
|
561 |
{
|
sl@0
|
562 |
return iRealBuf->ReadL(aPtr,aMaxLength);
|
sl@0
|
563 |
}
|
sl@0
|
564 |
|
sl@0
|
565 |
|
sl@0
|
566 |
TInt HDictionaryStoreBuf::DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
|
sl@0
|
567 |
{
|
sl@0
|
568 |
return iRealBuf->ReadL(aDes,aMaxLength,aStatus);
|
sl@0
|
569 |
}
|
sl@0
|
570 |
|
sl@0
|
571 |
|
sl@0
|
572 |
TStreamTransfer HDictionaryStoreBuf::DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer)
|
sl@0
|
573 |
{
|
sl@0
|
574 |
return iRealBuf->ReadL(anInput,aTransfer);
|
sl@0
|
575 |
}
|
sl@0
|
576 |
|
sl@0
|
577 |
|
sl@0
|
578 |
void HDictionaryStoreBuf::DoWriteL(const TAny* aPtr,TInt aLength)
|
sl@0
|
579 |
{
|
sl@0
|
580 |
iRealBuf->WriteL(aPtr,aLength);
|
sl@0
|
581 |
}
|
sl@0
|
582 |
|
sl@0
|
583 |
|
sl@0
|
584 |
TInt HDictionaryStoreBuf::DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
|
sl@0
|
585 |
{
|
sl@0
|
586 |
return iRealBuf->WriteL(aDes,aMaxLength,aStatus);
|
sl@0
|
587 |
}
|
sl@0
|
588 |
|
sl@0
|
589 |
|
sl@0
|
590 |
TStreamTransfer HDictionaryStoreBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
|
sl@0
|
591 |
{
|
sl@0
|
592 |
return iRealBuf->WriteL(anOutput,aTransfer);
|
sl@0
|
593 |
}
|
sl@0
|
594 |
|
sl@0
|
595 |
|
sl@0
|
596 |
TStreamPos HDictionaryStoreBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
|
sl@0
|
597 |
{
|
sl@0
|
598 |
return iRealBuf->SeekL(aMark,aLocation,anOffset);
|
sl@0
|
599 |
}
|
sl@0
|
600 |
|
sl@0
|
601 |
// class HNullBuf
|
sl@0
|
602 |
|
sl@0
|
603 |
HNullBuf::HNullBuf()
|
sl@0
|
604 |
{
|
sl@0
|
605 |
Set(0,0,ERead);
|
sl@0
|
606 |
}
|
sl@0
|
607 |
|
sl@0
|
608 |
void HNullBuf::DoRelease()
|
sl@0
|
609 |
{
|
sl@0
|
610 |
delete this;
|
sl@0
|
611 |
}
|
sl@0
|
612 |
|