Update contrib.
1 // Copyright (c) 2006-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
21 #include "BaSsndStore.h"
22 #include "BASSNDUID.h"
24 // The repository format is: one setting for MajorUID, one
25 // setting for sound type, one setting for the actual sound
26 // data, and then leave one key space empty to round up to
28 const TUint32 KSoundCatPartialKey = 0x1;
29 const TUint32 KSoundTypePartialKey = 0x2;
30 const TUint32 KSoundSettingPartialKey = 0x3;
31 const TUint32 KNumUnusedKeySpace = 1;
32 const TUint32 KSsndKeyMask = 0x3;
34 const TUint32 KKeyOffsetFromSoundType =
35 KSoundSettingPartialKey - KSoundTypePartialKey;
36 const TUint32 KKeyOffsetFromSoundCat =
37 KSoundSettingPartialKey - KSoundCatPartialKey;
38 const TUint32 KSsndBlockSize = KSoundSettingPartialKey + KNumUnusedKeySpace;
40 /** static factory method to instantiate an instance of CBaSsndStore */
41 CBaSsndStore* CBaSsndStore::NewL()
43 CBaSsndStore* self = new(ELeave)CBaSsndStore;
44 CleanupStack::PushL(self);
50 /** CBaSsndStore constructor */
51 CBaSsndStore::CBaSsndStore()
55 /** standard two phase construction to setup the CBaSsndStore object */
56 void CBaSsndStore::ConstructL()
58 iRepository = CRepository::NewL(KSystemSoundRepositoryUID);
61 /** CBaSsndStore destructor */
62 CBaSsndStore::~CBaSsndStore()
67 /** Retrieve sound data from repository.
68 @param aSoundType identifies the sound to retrieve.
69 @param aInfo contains the sound data on return.
70 @leave any of the system-wide error codes.
72 void CBaSsndStore::GetSoundL(const TBaSystemSoundType& aSoundType,
73 TBaSystemSoundInfo& aInfo) const
76 FindKeyL(aSoundType, key, EFalse);
78 HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
79 TPtr8 buf8 = hbuf8->Des();
80 User::LeaveIfError(iRepository->Get(key + KKeyOffsetFromSoundType, buf8));
82 RDesReadStream strm(buf8);
83 aInfo.InternalizeL(strm);
85 CleanupStack::PopAndDestroy(hbuf8);
87 if (! (aSoundType == aInfo.iType))
89 User::Leave(KErrCorrupt);
93 /** Search for a sound from storage and return the sound data.
94 @param aInfo as input contains the ID of the sound instance to
95 get. As output, return the sound info if operation successful.
96 @return KErrNone if successful, otherwise KErrNotFound or
97 any other system-wide errors.
99 TInt CBaSsndStore::GetSound(const TBaSystemSoundType& aSoundType,
100 TBaSystemSoundInfo& aInfo) const
102 TRAPD(err, GetSoundL(aSoundType, aInfo));
107 Overwrite existing sound if it is already in sound table.
108 Add it if it does not exist.
109 @param aInfo the sound data to save in repository.
110 @leave one of the system-wide errors if set fails.
112 void CBaSsndStore::SetSoundL(const TBaSystemSoundInfo& aInfo) const
114 HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
115 TPtr8 buf8 = hbuf8->Des();
116 RDesWriteStream writeStream( buf8 );
117 aInfo.ExternalizeL(writeStream);
118 writeStream.CommitL();
120 TBaSystemSoundType ssType(aInfo.iType);
121 // If two threads simultaneously add sound, one of them
122 // will fail. Hence retry.
123 const TInt KMaxRetryTransaction = 3;
124 TInt err(KErrLocked);
126 (i < KMaxRetryTransaction) && (err == KErrLocked || err == KErrAbort);
129 TRAP(err, SetSoundInTransactionL(ssType, buf8));
132 CleanupStack::PopAndDestroy(hbuf8);
133 User::LeaveIfError(err);
136 /** If the given sound type exists in repository it is over written.
137 If not exists, create a new entry. This modify/create logic must be
138 wrapped in a CentRep transaction.
139 @param aSoundType Use this unique indentifier to find
140 if the sound already exists in repository. If it exists
141 use Set. If not, use Create.
142 @param aDes8 contains the sound data streamed to a TDesC8 buffer.
143 @leave any of the system-wide errors.
145 void CBaSsndStore::SetSoundInTransactionL(TBaSystemSoundType& aSoundType,
146 const TDesC8& aDes8) const
148 User::LeaveIfError( iRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction) );
149 iRepository->CleanupCancelTransactionPushL();
154 KeyOfSoundTypeL(aSoundType, key);
156 if (key != NCentralRepositoryConstants::KUnspecifiedKey)
158 User::LeaveIfError(iRepository->Set(key + KKeyOffsetFromSoundType, aDes8));
159 User::LeaveIfError(iRepository->CommitTransaction(errorId));
160 CleanupStack::Pop(); // transaction
164 FindNextEmptySlotL(key);
166 TPckg<TBaSystemSoundUid> ssuid(aSoundType.iMajor);
167 User::LeaveIfError(iRepository->Create(key++, ssuid));
169 TPckg<TBaSystemSoundType> sstype(aSoundType);
170 User::LeaveIfError(iRepository->Create(key++, sstype));
172 User::LeaveIfError(iRepository->Create(key, aDes8) );
174 User::LeaveIfError(iRepository->CommitTransaction(errorId));
175 CleanupStack::Pop(); // transaction
178 /** Get all instances of a sound category
179 @param aSSUid identifies the category to retrieve.
180 @param aArray output parameter to return the sound instances
181 @return KErrNone if successful, else one of the system-wide error codes
183 TInt CBaSsndStore::GetSoundCategory(const TBaSystemSoundUid& aSSUid,
184 CArrayFixFlat<TBaSystemSoundInfo>& aArray) const
186 TRAPD(ret, GetSoundCategoryL(aSSUid, aArray));
190 /** Get all instances of a sound category.
191 @param aSSUid identifies the category to retrieve.
192 @param aArray output parameter to return the sound instances.
193 @leave any of the system-wide error codes.
195 void CBaSsndStore::GetSoundCategoryL(const TBaSystemSoundUid& aSSUid,
196 CArrayFixFlat<TBaSystemSoundInfo>& aArray) const
198 RArray<TUint32> keys;
199 FindKeyL(aSSUid, keys);
200 CleanupClosePushL(keys);
202 HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
203 TPtr8 buf8 = hbuf8->Des();
204 TBaSystemSoundInfo* sound = new(ELeave) TBaSystemSoundInfo;
205 CleanupStack::PushL(sound);
207 TInt n = keys.Count();
208 for (TInt i = 0; i < n; i++)
210 User::LeaveIfError(iRepository->Get(
211 keys[i] + KKeyOffsetFromSoundCat, buf8));
212 RDesReadStream strm(buf8);
213 sound->InternalizeL(strm);
214 aArray.AppendL(*sound);
217 CleanupStack::PopAndDestroy(sound);
218 CleanupStack::PopAndDestroy(hbuf8);
219 CleanupStack::PopAndDestroy(&keys);
223 Search for a sound instance in repository.
224 @param aSoundType identifies the TBaSystemSoundType to search for.
225 @param aKey output parameter containing the key of the sound type.
226 @param aNoLeaveIfNotFound indicate whether this method should leave if
227 the sound type is not found.
228 @leave KErrNotFound if not found, plus other system-wide errors.
230 void CBaSsndStore::FindKeyL(const TBaSystemSoundType& aSoundType,
232 TBool aNoLeaveIfNotFound) const
234 aKey = NCentralRepositoryConstants::KUnspecifiedKey;
236 TBaSystemSoundType soundTypeCopy = aSoundType;
237 TPckg<TBaSystemSoundType> target(soundTypeCopy);
238 RArray<TUint32> foundIds;
240 TInt ret = iRepository->FindEqL(KSoundTypePartialKey,
241 KSsndKeyMask, target, foundIds);
242 if (ret == KErrNotFound && aNoLeaveIfNotFound)
247 User::LeaveIfError(ret);
253 Search for a sound category
254 @param aSSUid the sound category to search for.
255 @param aKeys output parameter containing keys having the sound category
257 @leave KErrNotFound if not found, plus other system-wide errors.
259 void CBaSsndStore::FindKeyL(const TBaSystemSoundUid& aSSUid, RArray<TUint32>& aKeys) const
261 TBaSystemSoundUid ssuidCopy = aSSUid;
262 TPckg<TBaSystemSoundUid> target(ssuidCopy);
264 User::LeaveIfError( iRepository->FindEqL(KSoundCatPartialKey,
265 KSsndKeyMask, target, aKeys) );
268 /** Call FindKeyL with aNoLeaveIfNotFound parameter set to true.
269 Used in SetSound to intercept and save the key.
270 @param aSoundType is the sound type to search for.
271 @param aKey output parameter to hold the key of the sound type.
272 @see CBaSsndStore::FindKeyL
274 void CBaSsndStore::KeyOfSoundTypeL(TBaSystemSoundType& aSoundType,
277 FindKeyL(aSoundType, aKey, ETrue);
281 Find the next unused key to append a new sound.
282 @param aKey on return contains the key to use to store
284 @leave any of the system-wide error codes.
286 void CBaSsndStore::FindNextEmptySlotL(TUint32& aKey) const
288 RArray<TUint32> foundIds;
290 TInt ret = iRepository->FindL(KSoundSettingPartialKey,
291 KSsndKeyMask, foundIds);
292 if (ret == KErrNotFound)
294 // Empty repository. Start storing at key 1
297 else if (ret == KErrNone)
299 // The array is sorted. The max key is the last one in array.
300 TInt n = foundIds.Count();
301 TUint32 maxkey = foundIds[n-1];
304 TUint32 expectedArraySize = maxkey / KSsndBlockSize + 1;
305 if (expectedArraySize > n)
307 // Rogue app bypassed BaSystemSound and added non-contiguous
308 // entries in repository.
309 User::Leave(KErrCorrupt);
312 // next record to be stored at maxkey + 2
313 aKey = maxkey + KNumUnusedKeySpace + 1;