os/ossrv/lowlevellibsandfws/apputils/src/BaSsndStore.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2006-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
/**
sl@0
    17
 @file
sl@0
    18
 @internalTechnology
sl@0
    19
*/
sl@0
    20
#include <s32mem.h>
sl@0
    21
#include "BaSsndStore.h"
sl@0
    22
#include "BASSNDUID.h"
sl@0
    23
sl@0
    24
// The repository format is: one setting for MajorUID, one
sl@0
    25
// setting for sound type, one setting for the actual sound
sl@0
    26
// data, and then leave one key space empty to round up to
sl@0
    27
// power of 2.
sl@0
    28
const TUint32 KSoundCatPartialKey = 0x1;
sl@0
    29
const TUint32 KSoundTypePartialKey = 0x2;
sl@0
    30
const TUint32 KSoundSettingPartialKey = 0x3;
sl@0
    31
const TUint32 KNumUnusedKeySpace = 1;
sl@0
    32
const TUint32 KSsndKeyMask = 0x3;
sl@0
    33
sl@0
    34
const TUint32 KKeyOffsetFromSoundType = 
sl@0
    35
	KSoundSettingPartialKey - KSoundTypePartialKey;
sl@0
    36
const TUint32 KKeyOffsetFromSoundCat = 
sl@0
    37
	KSoundSettingPartialKey - KSoundCatPartialKey;
sl@0
    38
const TUint32 KSsndBlockSize = KSoundSettingPartialKey + KNumUnusedKeySpace;
sl@0
    39
sl@0
    40
/** static factory method to instantiate an instance of CBaSsndStore */
sl@0
    41
CBaSsndStore* CBaSsndStore::NewL()
sl@0
    42
	{
sl@0
    43
	CBaSsndStore* self = new(ELeave)CBaSsndStore;
sl@0
    44
	CleanupStack::PushL(self);
sl@0
    45
	self->ConstructL();
sl@0
    46
	CleanupStack::Pop();
sl@0
    47
	return self;
sl@0
    48
	}
sl@0
    49
sl@0
    50
/** CBaSsndStore constructor */
sl@0
    51
CBaSsndStore::CBaSsndStore()
sl@0
    52
	{
sl@0
    53
	}
sl@0
    54
sl@0
    55
/** standard two phase construction to setup the CBaSsndStore object */
sl@0
    56
void CBaSsndStore::ConstructL()
sl@0
    57
	{
sl@0
    58
	iRepository = CRepository::NewL(KSystemSoundRepositoryUID);
sl@0
    59
	}
sl@0
    60
sl@0
    61
/** CBaSsndStore destructor */
sl@0
    62
CBaSsndStore::~CBaSsndStore()
sl@0
    63
	{
sl@0
    64
	delete iRepository;
sl@0
    65
	}
sl@0
    66
sl@0
    67
/** Retrieve sound data from repository.
sl@0
    68
@param aSoundType identifies the sound to retrieve.
sl@0
    69
@param aInfo contains the sound data on return.
sl@0
    70
@leave any of the system-wide error codes.
sl@0
    71
*/
sl@0
    72
void CBaSsndStore::GetSoundL(const TBaSystemSoundType& aSoundType,
sl@0
    73
							 TBaSystemSoundInfo& aInfo) const
sl@0
    74
	{
sl@0
    75
	TUint32 key;
sl@0
    76
	FindKeyL(aSoundType, key, EFalse);
sl@0
    77
sl@0
    78
	HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
sl@0
    79
	TPtr8 buf8 = hbuf8->Des();
sl@0
    80
	User::LeaveIfError(iRepository->Get(key + KKeyOffsetFromSoundType, buf8));
sl@0
    81
sl@0
    82
	RDesReadStream strm(buf8);
sl@0
    83
	aInfo.InternalizeL(strm);
sl@0
    84
sl@0
    85
	CleanupStack::PopAndDestroy(hbuf8);
sl@0
    86
sl@0
    87
	if (! (aSoundType == aInfo.iType))
sl@0
    88
		{
sl@0
    89
		User::Leave(KErrCorrupt);
sl@0
    90
		}
sl@0
    91
	}
sl@0
    92
sl@0
    93
/** Search for a sound from storage and return the sound data.
sl@0
    94
@param aInfo as input contains the ID of the sound instance to
sl@0
    95
	get. As output, return the sound info if operation successful. 
sl@0
    96
@return KErrNone if successful, otherwise KErrNotFound or
sl@0
    97
	any other system-wide errors.
sl@0
    98
*/
sl@0
    99
TInt CBaSsndStore::GetSound(const TBaSystemSoundType& aSoundType,
sl@0
   100
							TBaSystemSoundInfo& aInfo) const
sl@0
   101
	{
sl@0
   102
	TRAPD(err, GetSoundL(aSoundType, aInfo));
sl@0
   103
	return err;
sl@0
   104
	}
sl@0
   105
sl@0
   106
/**
sl@0
   107
Overwrite existing sound if it is already in sound table.
sl@0
   108
Add it if it does not exist.
sl@0
   109
@param aInfo the sound data to save in repository.
sl@0
   110
@leave one of the system-wide errors if set fails.
sl@0
   111
*/
sl@0
   112
void CBaSsndStore::SetSoundL(const TBaSystemSoundInfo& aInfo) const
sl@0
   113
	{
sl@0
   114
	HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
sl@0
   115
	TPtr8 buf8 = hbuf8->Des();
sl@0
   116
	RDesWriteStream writeStream( buf8 );
sl@0
   117
	aInfo.ExternalizeL(writeStream);
sl@0
   118
	writeStream.CommitL();
sl@0
   119
sl@0
   120
	TBaSystemSoundType ssType(aInfo.iType);
sl@0
   121
	// If two threads simultaneously add sound, one of them
sl@0
   122
	// will fail. Hence retry.
sl@0
   123
	const TInt KMaxRetryTransaction = 3;
sl@0
   124
	TInt err(KErrLocked);
sl@0
   125
	for (TInt i = 0; 
sl@0
   126
		 (i < KMaxRetryTransaction) && (err == KErrLocked || err == KErrAbort);
sl@0
   127
		 i++)
sl@0
   128
		{
sl@0
   129
		TRAP(err, SetSoundInTransactionL(ssType, buf8));
sl@0
   130
		}
sl@0
   131
sl@0
   132
	CleanupStack::PopAndDestroy(hbuf8);
sl@0
   133
	User::LeaveIfError(err);
sl@0
   134
	}
sl@0
   135
sl@0
   136
/** If the given sound type exists in repository it is over written.
sl@0
   137
If not exists, create a new entry. This modify/create logic must be
sl@0
   138
wrapped in a CentRep transaction.
sl@0
   139
@param aSoundType Use this unique indentifier to find
sl@0
   140
       if the sound already exists in repository. If it exists
sl@0
   141
	   use Set. If not, use Create.
sl@0
   142
@param aDes8 contains the sound data streamed to a TDesC8 buffer.
sl@0
   143
@leave any of the system-wide errors.
sl@0
   144
*/
sl@0
   145
void CBaSsndStore::SetSoundInTransactionL(TBaSystemSoundType& aSoundType,
sl@0
   146
										  const TDesC8& aDes8) const
sl@0
   147
	{
sl@0
   148
	User::LeaveIfError( iRepository->StartTransaction(CRepository::EConcurrentReadWriteTransaction) );
sl@0
   149
	iRepository->CleanupCancelTransactionPushL();
sl@0
   150
sl@0
   151
	TUint32 errorId;
sl@0
   152
sl@0
   153
	TUint32 key;
sl@0
   154
	KeyOfSoundTypeL(aSoundType, key);
sl@0
   155
sl@0
   156
	if (key != NCentralRepositoryConstants::KUnspecifiedKey)
sl@0
   157
		{
sl@0
   158
		User::LeaveIfError(iRepository->Set(key + KKeyOffsetFromSoundType, aDes8));
sl@0
   159
		User::LeaveIfError(iRepository->CommitTransaction(errorId));
sl@0
   160
		CleanupStack::Pop(); // transaction
sl@0
   161
		return;
sl@0
   162
		}
sl@0
   163
	
sl@0
   164
	FindNextEmptySlotL(key);
sl@0
   165
sl@0
   166
	TPckg<TBaSystemSoundUid> ssuid(aSoundType.iMajor);
sl@0
   167
	User::LeaveIfError(iRepository->Create(key++, ssuid));
sl@0
   168
sl@0
   169
	TPckg<TBaSystemSoundType> sstype(aSoundType);
sl@0
   170
	User::LeaveIfError(iRepository->Create(key++, sstype));
sl@0
   171
sl@0
   172
	User::LeaveIfError(iRepository->Create(key, aDes8) );
sl@0
   173
sl@0
   174
	User::LeaveIfError(iRepository->CommitTransaction(errorId));
sl@0
   175
	CleanupStack::Pop(); // transaction
sl@0
   176
	}
sl@0
   177
sl@0
   178
/** Get all instances of a sound category
sl@0
   179
@param aSSUid identifies the category to retrieve.
sl@0
   180
@param aArray output parameter to return the sound instances
sl@0
   181
@return KErrNone if successful, else one of the system-wide error codes
sl@0
   182
*/
sl@0
   183
TInt CBaSsndStore::GetSoundCategory(const TBaSystemSoundUid& aSSUid,
sl@0
   184
									CArrayFixFlat<TBaSystemSoundInfo>& aArray) const
sl@0
   185
	{
sl@0
   186
	TRAPD(ret, GetSoundCategoryL(aSSUid, aArray));
sl@0
   187
	return ret;
sl@0
   188
	}
sl@0
   189
sl@0
   190
/** Get all instances of a sound category.
sl@0
   191
@param aSSUid identifies the category to retrieve.
sl@0
   192
@param aArray output parameter to return the sound instances.
sl@0
   193
@leave any of the system-wide error codes.
sl@0
   194
*/
sl@0
   195
void CBaSsndStore::GetSoundCategoryL(const TBaSystemSoundUid& aSSUid,
sl@0
   196
									 CArrayFixFlat<TBaSystemSoundInfo>& aArray) const
sl@0
   197
	{
sl@0
   198
	RArray<TUint32> keys;
sl@0
   199
	FindKeyL(aSSUid, keys);
sl@0
   200
	CleanupClosePushL(keys);
sl@0
   201
sl@0
   202
	HBufC8* hbuf8 = HBufC8::NewLC(sizeof(TBaSystemSoundInfo));
sl@0
   203
	TPtr8 buf8 = hbuf8->Des();
sl@0
   204
	TBaSystemSoundInfo* sound = new(ELeave) TBaSystemSoundInfo;
sl@0
   205
	CleanupStack::PushL(sound);
sl@0
   206
sl@0
   207
	TInt n = keys.Count();
sl@0
   208
	for (TInt i = 0; i < n; i++)
sl@0
   209
		{
sl@0
   210
		User::LeaveIfError(iRepository->Get(
sl@0
   211
			keys[i] + KKeyOffsetFromSoundCat, buf8));
sl@0
   212
		RDesReadStream strm(buf8);
sl@0
   213
		sound->InternalizeL(strm);
sl@0
   214
		aArray.AppendL(*sound);
sl@0
   215
		}
sl@0
   216
sl@0
   217
	CleanupStack::PopAndDestroy(sound);
sl@0
   218
	CleanupStack::PopAndDestroy(hbuf8);
sl@0
   219
	CleanupStack::PopAndDestroy(&keys);
sl@0
   220
	}
sl@0
   221
sl@0
   222
/**
sl@0
   223
Search for a sound instance in repository.
sl@0
   224
@param aSoundType identifies the TBaSystemSoundType to search for.
sl@0
   225
@param aKey output parameter containing the key of the sound type.
sl@0
   226
@param aNoLeaveIfNotFound indicate whether this method should leave if
sl@0
   227
		the sound type is not found.
sl@0
   228
@leave KErrNotFound if not found, plus other system-wide errors.
sl@0
   229
*/
sl@0
   230
void CBaSsndStore::FindKeyL(const TBaSystemSoundType& aSoundType,
sl@0
   231
							TUint32& aKey,
sl@0
   232
							TBool aNoLeaveIfNotFound) const
sl@0
   233
	{
sl@0
   234
	aKey = NCentralRepositoryConstants::KUnspecifiedKey;
sl@0
   235
sl@0
   236
	TBaSystemSoundType soundTypeCopy = aSoundType;
sl@0
   237
	TPckg<TBaSystemSoundType> target(soundTypeCopy);
sl@0
   238
	RArray<TUint32> foundIds;
sl@0
   239
sl@0
   240
	TInt ret = iRepository->FindEqL(KSoundTypePartialKey, 
sl@0
   241
		KSsndKeyMask, target, foundIds);
sl@0
   242
	if (ret == KErrNotFound && aNoLeaveIfNotFound)
sl@0
   243
		{
sl@0
   244
		return;
sl@0
   245
		}
sl@0
   246
sl@0
   247
	User::LeaveIfError(ret);
sl@0
   248
	aKey = foundIds[0];
sl@0
   249
	foundIds.Reset();
sl@0
   250
	}
sl@0
   251
sl@0
   252
/**
sl@0
   253
Search for a sound category
sl@0
   254
@param aSSUid the sound category to search for.
sl@0
   255
@param aKeys output parameter containing keys having the sound category
sl@0
   256
		as their values.
sl@0
   257
@leave KErrNotFound if not found, plus other system-wide errors.
sl@0
   258
*/
sl@0
   259
void CBaSsndStore::FindKeyL(const TBaSystemSoundUid& aSSUid, RArray<TUint32>& aKeys) const
sl@0
   260
	{
sl@0
   261
	TBaSystemSoundUid ssuidCopy = aSSUid;
sl@0
   262
	TPckg<TBaSystemSoundUid> target(ssuidCopy);
sl@0
   263
sl@0
   264
	User::LeaveIfError( iRepository->FindEqL(KSoundCatPartialKey, 
sl@0
   265
		KSsndKeyMask, target, aKeys) );
sl@0
   266
	}
sl@0
   267
sl@0
   268
/** Call FindKeyL with aNoLeaveIfNotFound parameter set to true.
sl@0
   269
Used in SetSound to intercept and save the key.
sl@0
   270
@param aSoundType is the sound type to search for.
sl@0
   271
@param aKey output parameter to hold the key of the sound type.
sl@0
   272
@see CBaSsndStore::FindKeyL
sl@0
   273
*/
sl@0
   274
void CBaSsndStore::KeyOfSoundTypeL(TBaSystemSoundType& aSoundType,
sl@0
   275
								   TUint32& aKey) const
sl@0
   276
	{
sl@0
   277
	FindKeyL(aSoundType, aKey, ETrue);
sl@0
   278
	}
sl@0
   279
sl@0
   280
/**
sl@0
   281
Find the next unused key to append a new sound.
sl@0
   282
@param aKey on return contains the key to use to store
sl@0
   283
	the next sound.
sl@0
   284
@leave any of the system-wide error codes.
sl@0
   285
*/
sl@0
   286
void CBaSsndStore::FindNextEmptySlotL(TUint32& aKey) const
sl@0
   287
	{
sl@0
   288
	RArray<TUint32> foundIds;
sl@0
   289
sl@0
   290
	TInt ret = iRepository->FindL(KSoundSettingPartialKey,
sl@0
   291
		KSsndKeyMask, foundIds);
sl@0
   292
	if (ret == KErrNotFound)
sl@0
   293
		{
sl@0
   294
		// Empty repository. Start storing at key 1
sl@0
   295
		aKey = 1;
sl@0
   296
		}
sl@0
   297
	else if (ret == KErrNone)
sl@0
   298
		{
sl@0
   299
	    // The array is sorted. The max key is the last one in array.
sl@0
   300
		TInt n = foundIds.Count();
sl@0
   301
		TUint32 maxkey = foundIds[n-1];
sl@0
   302
		foundIds.Reset();
sl@0
   303
sl@0
   304
		TUint32 expectedArraySize = maxkey / KSsndBlockSize + 1;
sl@0
   305
		if (expectedArraySize > n)
sl@0
   306
			{
sl@0
   307
			// Rogue app bypassed BaSystemSound and added non-contiguous
sl@0
   308
			// entries in repository.
sl@0
   309
			User::Leave(KErrCorrupt);
sl@0
   310
			}
sl@0
   311
sl@0
   312
		// next record to be stored at maxkey + 2
sl@0
   313
		aKey = maxkey + KNumUnusedKeySpace + 1;
sl@0
   314
		}
sl@0
   315
	else
sl@0
   316
		{
sl@0
   317
		User::Leave(ret);
sl@0
   318
		}
sl@0
   319
	}