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.
19 #define UNUSED_VAR(a) a = a
21 const TInt KStreamDictGranularity=16;
23 NONSHARABLE_CLASS(HDictionaryStoreBuf) : public MStreamBuf
26 HDictionaryStoreBuf(MStreamBuf& aBuf,CDictionaryStore& aDic,TStreamId aStreamId,TUid aUid);
29 void DoSynchL(); // only update the stream dic during synch
30 // just delegate the rest straight through...
32 TInt DoReadL(TAny* aPtr,TInt aMaxLength);
33 TInt DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus);
34 TStreamTransfer DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer);
35 void DoWriteL(const TAny* aPtr,TInt aLength);
36 TInt DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus);
37 TStreamTransfer DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer);
38 TStreamPos DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset);
40 MStreamBuf* iRealBuf; // delegate everything to this
41 CDictionaryStore* iDictionary;
42 TStreamId iStreamId; // id of the actual stream in the store
43 TUid iStreamUid; // Uid to be paired with the stream id in the dic
46 NONSHARABLE_CLASS(HNullBuf) : public TMemBuf
54 ////////////////////////////////////////
56 ////////////////////////////////////////
58 EXPORT_C CStreamDictionary* CStreamDictionary::NewL()
59 /** Allocates, constructs and returns a pointer to a new stream dictionary.
61 The function leaves if it cannot complete successfully.
63 @return A pointer to the new stream dictionary object. */
65 return new(ELeave) CStreamDictionary;
68 EXPORT_C CStreamDictionary* CStreamDictionary::NewLC()
69 /** Allocates, constructs and returns a pointer to a new stream dictionary, putting
70 the pointer onto the cleanup stack.
72 Putting the pointer onto the cleanup stack allows the object and allocated
73 resources to be cleaned up if a subsequent leave occurs.
75 The function leaves if it cannot complete successfully.
77 @return A pointer to the new stream dictionary object. */
79 CStreamDictionary* dict=NewL();
80 CleanupStack::PushL(dict);
84 EXPORT_C CStreamDictionary::CStreamDictionary()
85 : iArray(KStreamDictGranularity)
88 EXPORT_C CStreamDictionary::~CStreamDictionary()
89 /** Frees resources owned by the object, prior to its destruction. */
92 EXPORT_C void CStreamDictionary::AssignL(TUid aUid,TStreamId anId)
93 /** Creates or changes an association between a UID and a stream id.
95 If the stream dictionary already contains an entry with the same UID as aUid,
96 then the associated stream id in that entry is replaced by anId.
98 If anId has the value KNullStreamId, then the function attempts to remove
99 the entry whose unique identifier matches aUid.
102 @param anId The stream id.
103 @see CStreamDictionary::Remove() */
105 if (anId==KNullStreamId)
107 Remove(aUid); // default associated stream is null
113 TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
115 if (iArray.Find(entry,key,i)==0)
121 __ASSERT_DEBUG(i==iArray.Count(),User::Invariant());
123 iArray.InsertL(i,entry);
126 EXPORT_C void CStreamDictionary::Remove(TUid aUid)
127 /** Removes an association from the stream dictionary.
129 The function searches the stream dictionary for an entry whose UID matches
130 aUid. If a match is found, the entry is removed. If no match is found, the
131 stream dictionary remains unchanged.
133 @param aUid The UID whose matching entry is to be removed from the stream
138 TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
140 if (iArray.Find(entry,key,i)==0)
144 EXPORT_C TStreamId CStreamDictionary::At(TUid aUid) const
145 /** Returns the stream id associated with the specified UID.
147 @param aUid The UID whose associated stream id is to be returned.
148 @return The stream id associated with the specified UID.KNullStreamId, if no
149 entry in the stream dictionary contains the specified UID. */
153 TKeyArrayFix key(_FOFF(TEntry,iUid),ECmpTUint32);
155 if (iArray.Find(entry,key,i)!=0)
156 return KNullStreamId;
158 return iArray[i].iId;
161 EXPORT_C TBool CStreamDictionary::IsNull() const
162 /** Tests whether the stream dictionary is empty.
164 @return True, if the stream dictionary is empty. False if the stream dictionary
165 contains at least one entry. */
167 return iArray.Count()==0;
170 EXPORT_C void CStreamDictionary::ExternalizeL(RWriteStream& aStream) const
171 /** Externalises an object of this class to a write stream.
173 The presence of this function means that the standard templated operator<<()
174 can be used to externalise objects of this class.
176 @param aStream Stream to which the object should be externalised */
181 EXPORT_C void CStreamDictionary::InternalizeL(RReadStream& aStream)
182 /** Internalises an object of this class from a read stream.
184 The presence of this function means that the standard templated operator>>()
185 can be used to internalise objects of this class.
187 Note that this function has assignment semantics. It replaces the old value
188 of the object with a new value read from the read stream.
190 @param aStream Stream from which the object should be internalised. */
195 void CStreamDictionary::TEntry::ExternalizeL(RWriteStream& aStream) const
201 void CStreamDictionary::TEntry::InternalizeL(RReadStream& aStream)
208 ////////////////////////////////////////
210 ////////////////////////////////////////
212 EXPORT_C CDictionaryStore::~CDictionaryStore()
213 /** Frees resources owned by the object, prior to its destruction. */
220 EXPORT_C void CDictionaryStore::ConstructL()
222 iDictionary = CStreamDictionary::NewL();
223 RStoreWriteStream stream;
224 iStore->SetRootL(stream.CreateLC(*iStore));
225 stream<<*iDictionary;
227 CleanupStack::PopAndDestroy(); // dictionary stream
232 EXPORT_C void CDictionaryStore::Remove(TUid aUid)
233 /** Removes the stream associated with the specified UID from the dictionary store.
235 If there is no stream associated with the specified UID, then the dictionary
236 store remains unchanged.
238 The function cannot leave; it returns whether or not it is succesful.
240 @param aUid The UID whose associated stream is to be removed from the dictionary
243 TRAPD(ignore,RemoveL(aUid));
248 EXPORT_C void CDictionaryStore::RemoveL(TUid aUid)
249 /** Removes the stream associated with the specified UID from the dictionary store
250 and leaves if unsuccessful.
252 If there is no stream associated with the specified UID, then the dictionary
253 store remains unchanged.
255 @param aUid The UID whose associated stream is to be removed from the dictionary
258 TStreamId id = DictionaryL()->At(aUid);
259 if (id!=KNullStreamId)
261 DictionaryL()->Remove(aUid);
263 iDictionaryHasChanged=ETrue;
268 EXPORT_C TInt CDictionaryStore::Commit()
271 It establishes a new commit point and then compacts the dictionary store.
272 Typically, this is done after changes to new or existing streams are complete
273 and the streams themselves have been committed.
275 Establishing a new commit point makes changes to the store permanent. Until
276 such changes are committed, they can be rolled back, effectively causing the
277 store to revert back to its state before the changes were made.
279 This ensures that persistent data moves from one consistent state to another
280 and guarantees the integrity of persistent store data in the event of failures.
281 In particular, if a process terminates or a media failure occurs, the store
282 reverts automatically to its state at the last successful commit point.
284 @return KErrNone if successful, otherwise another of the system-wide error
287 TRAPD(ret,CommitL());
292 EXPORT_C void CDictionaryStore::CommitL()
293 /** Commits changes and leaves if unsuccessful.
295 @see CDictionaryStore::Commit() */
297 if (iDictionaryHasChanged)
299 // rewrite the root stream
300 RStoreWriteStream stream;
301 stream.ReplaceLC(*iStore,iStore->Root());
302 stream<< *DictionaryL();
304 CleanupStack::PopAndDestroy(); // dictionary stream
308 // compact the store, ignoring any failure (commit was good)
312 if (iStore->ReclaimL() >= KDictionaryCommitThreshold)
322 EXPORT_C void CDictionaryStore::Revert()
323 /** Rolls the dictionary store back to its state at the last commit point.
325 A commit point is set using the Commit() or CommitL() functions.
327 The function returns, whether or not it completes successfully.
329 @see CDictionaryStore::Commit()
330 @see CDictionaryStore::CommitL() */
332 TRAPD(ignore,RevertL());
337 EXPORT_C void CDictionaryStore::RevertL()
338 /** Rolls the dictionary store back to its state at the last commit point and leaves
341 A commit point is set using the Commit() or CommitL() functions.
343 @see CDictionaryStore::Commit()
344 @see CDictionaryStore::CommitL() */
348 iDictionaryHasChanged = EFalse;
353 CStreamDictionary* CDictionaryStore::DictionaryL() const
354 // Returns the handle of the stream dictionary, internalizing it first if necessary
358 // create a temporary dictionary
359 CStreamDictionary* dictionary = CStreamDictionary::NewLC();
360 // stream into the dictionary
361 RStoreReadStream root;
362 root.OpenLC(*iStore,iStore->Root());
365 CleanupStack::PopAndDestroy(); // root
367 MUTABLE_CAST(CStreamDictionary*&,iDictionary) = dictionary;
368 CleanupStack::Pop(); // dictionary
374 EXPORT_C TBool CDictionaryStore::IsNullL() const
375 /** Tests whether the dictionary stores stream dictionary is empty.
377 @return True, if the stream dictionary is empty. False, if the stream dictionary
378 contains at least one entry.
379 @see CStreamDictionary */
381 return DictionaryL()->IsNull();
385 EXPORT_C TBool CDictionaryStore::IsPresentL(TUid aUid) const
386 /** Tests whether the specified UID has an associated stream within this dictionary
390 @return True, if there is a stream associated with the specified UID in the
391 dictionary store; false otherwise.
392 @see CStreamDictionary */
394 return DictionaryL()->At(aUid)!=KNullStreamId;
398 MStreamBuf* CDictionaryStore::GetSourceL(TUid aUid) const
399 // Opens the stream matching aUid and returns its source.
400 // If no stream matches aUid an empty stream buf is returned
401 // Ownership of the source is transfered to the caller.
403 // get the id of the stream
404 TStreamId id = DictionaryL()->At(aUid);
405 // open the stream if it exists, otherwise open a dummy empty stream
406 if (id==KNullStreamId)
407 return (new(ELeave) HNullBuf);
410 // open the stream and steal its source
411 RStoreReadStream stream;
412 stream.OpenL(*iStore,id);
413 return stream.Source();
418 MStreamBuf* CDictionaryStore::GetSinkL(TUid aUid)
419 // If a stream matches aUid it is replaced ad its sink returned
420 // If no stream matches aUid a new stream is created and its sink returned
421 // Ownership of the sink is transfered to the caller.
423 // get the id of the stream
424 TStreamId id = DictionaryL()->At(aUid);
425 RStoreWriteStream stream;
426 if (id!=KNullStreamId)
428 stream.ReplaceLC(*iStore,id);
430 // create a new stream
431 id = stream.CreateLC(*iStore);
432 // create a dictionary store buffer to hold the incoming data
433 MStreamBuf* buf = new(ELeave) HDictionaryStoreBuf(*stream.Sink(),*this,id,aUid);
434 CleanupStack::Pop(); // stream;
439 //////////////////////////////////////////////
440 // RDictionaryReadStream
441 //////////////////////////////////////////////
443 EXPORT_C void RDictionaryReadStream::OpenL(const CDictionaryStore& aDictStore,TUid aUid)
444 // Open the stream if it exists, otherwise open a dummy empty stream
445 /** Opens the stream associated with the specified UID in the specified dictionary
446 store, and prepares it for reading.
448 @param aDictStore The dictionary store containing the stream.
449 @param aUid The unique identifier associated with the stream to be read. */
451 Attach(aDictStore.GetSourceL(aUid));
455 EXPORT_C void RDictionaryReadStream::OpenLC(const CDictionaryStore& aDictStore,TUid aUid)
456 /** Opens the stream associated with the specified UID in the specified dictionary
457 store, prepares it for reading, and puts a a cleanup item onto the cleanup
460 Placing a cleanup item on the cleanup stack allows allocated resources to
461 be cleaned up if a subsequent leave occurs.
463 @param aDictStore The dictionary store containing the stream.
464 @param aUid The unique identifier associated with the stream to be read. */
466 OpenL(aDictStore,aUid);
471 //////////////////////////////////////////////
472 // RDictionaryWriteStream
473 //////////////////////////////////////////////
475 EXPORT_C void RDictionaryWriteStream::AssignL(CDictionaryStore& aDictStore,TUid aUid)
476 // Replace the stream if it exists, otherwise create a new one
477 /** Prepares a stream in the specified dictionary store for writing.
479 If no stream is associated with the specified UID, then a new stream is created
480 and an association is made between the resulting stream id and the specified
481 UID. If a stream is currently associated with the specified UID, then this
482 existing stream is prepared for replacement.
484 @param aDictStore The dictionary store which contains the new or replacement
486 @param aUid The UID associated with the stream.
487 @see RWriteStream::Release() */
489 Attach(aDictStore.GetSinkL(aUid));
493 EXPORT_C void RDictionaryWriteStream::AssignLC(CDictionaryStore& aDictStore,TUid aUid)
494 /** Prepares a stream in the specified dictionary store for writing, and places
495 a cleanup item for this RDictionaryWriteStream object onto the cleanup stack.
497 If no stream is associated with the specified UID, then a new stream is created
498 and an association is made between the resulting stream id and the specified
499 UID. If a stream is currently associated with the specified UID, then this
500 existing stream is prepared for replacement.
502 Placing a cleanup item onto the cleanup stack allows allocated resources to
503 be cleaned up if a subsequent leave occurs.
505 @param aDictStore The dictionary store which contains the new or replacement
507 @param aUid The UID associated with the stream. */
509 AssignL(aDictStore,aUid);
514 //////////////////////////////////////////////
515 // HDictionaryStoreBuf
516 //////////////////////////////////////////////
518 HDictionaryStoreBuf::HDictionaryStoreBuf(MStreamBuf& aBuf,CDictionaryStore& aDic,TStreamId aStreamId,TUid aUid)
521 iStreamId(aStreamId),
526 void HDictionaryStoreBuf::DoSynchL()
527 // only update the stream dic during synch - this is when the stream gets added to the store
529 CStreamDictionary& dic = *iDictionary->DictionaryL();
530 TBool newEntry = (dic.At(iStreamUid)==KNullStreamId);
531 if (iRealBuf->SizeL()>0)
533 dic.AssignL(iStreamUid,iStreamId);
534 TInt ret = iRealBuf->Synch();
538 dic.Remove(iStreamUid);
542 iDictionary->iDictionaryHasChanged=ETrue;
547 iDictionary->iStore->Delete(iStreamId);
548 iDictionary->Remove(iStreamUid);
553 void HDictionaryStoreBuf::DoRelease()
560 TInt HDictionaryStoreBuf::DoReadL(TAny* aPtr,TInt aMaxLength)
562 return iRealBuf->ReadL(aPtr,aMaxLength);
566 TInt HDictionaryStoreBuf::DoReadL(TDes8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
568 return iRealBuf->ReadL(aDes,aMaxLength,aStatus);
572 TStreamTransfer HDictionaryStoreBuf::DoReadL(MStreamInput& anInput,TStreamTransfer aTransfer)
574 return iRealBuf->ReadL(anInput,aTransfer);
578 void HDictionaryStoreBuf::DoWriteL(const TAny* aPtr,TInt aLength)
580 iRealBuf->WriteL(aPtr,aLength);
584 TInt HDictionaryStoreBuf::DoWriteL(const TDesC8& aDes,TInt aMaxLength,TRequestStatus& aStatus)
586 return iRealBuf->WriteL(aDes,aMaxLength,aStatus);
590 TStreamTransfer HDictionaryStoreBuf::DoWriteL(MStreamOutput& anOutput,TStreamTransfer aTransfer)
592 return iRealBuf->WriteL(anOutput,aTransfer);
596 TStreamPos HDictionaryStoreBuf::DoSeekL(TMark aMark,TStreamLocation aLocation,TInt anOffset)
598 return iRealBuf->SeekL(aMark,aLocation,anOffset);
608 void HNullBuf::DoRelease()