os/kernelhwsrv/kernel/eka/include/drivers/soundsc.h
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 the License "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
// e32\include\drivers\soundsc.h
sl@0
    15
// Kernel side definitions for the shared chunk sound driver.
sl@0
    16
//
sl@0
    17
//
sl@0
    18
sl@0
    19
/**
sl@0
    20
 @file
sl@0
    21
 @internalTechnology
sl@0
    22
 @prototype
sl@0
    23
*/
sl@0
    24
sl@0
    25
#ifndef __SOUNDSC_H__
sl@0
    26
#define __SOUNDSC_H__
sl@0
    27
sl@0
    28
#include <d32soundsc.h>
sl@0
    29
#include <platform.h>
sl@0
    30
#include <kernel/kpower.h>
sl@0
    31
#include <e32ver.h>
sl@0
    32
sl@0
    33
/** The maximum number of client transfer requests which may be outstanding in a particular direction at any time. */
sl@0
    34
const TInt KMaxSndScRequestsPending=8;
sl@0
    35
sl@0
    36
/**
sl@0
    37
A bit-mask value for the sound configuration status variable DSoundScLdd::iSoundConfigFlags. This being set signifies
sl@0
    38
that the sound configuration has been set in hardware by the driver.
sl@0
    39
*/
sl@0
    40
const TUint KSndScSoundConfigIsSetup=0x00000001;
sl@0
    41
/**
sl@0
    42
A bit-mask value for the sound configuration status variable DSoundScLdd::iSoundConfigFlags. This being set signifies
sl@0
    43
that the record level / play volume has been set in hardware by the driver.
sl@0
    44
*/
sl@0
    45
const TUint KSndScVolumeIsSetup=0x00000002;
sl@0
    46
sl@0
    47
// Bit-mask values used when testing the driver
sl@0
    48
const TUint KSoundScTest_StartTransferError=0x01;
sl@0
    49
const TUint KSoundScTest_TransferDataError=0x02;
sl@0
    50
const TUint KSoundScTest_TransferTimeout=0x04;
sl@0
    51
sl@0
    52
// Forward declarations
sl@0
    53
class TAudioBuffer;
sl@0
    54
class DBufferManager;
sl@0
    55
class DSoundScLdd;
sl@0
    56
sl@0
    57
/**
sl@0
    58
@publishedPartner
sl@0
    59
@prototype
sl@0
    60
*/
sl@0
    61
class TSoundSharedChunkBufConfig : public TSharedChunkBufConfigBase
sl@0
    62
	{
sl@0
    63
public:
sl@0
    64
	/** The first entry of the buffer offset list. This list holds the offset from the start of the chunk
sl@0
    65
	for each buffer. This list is only valid if the flag KScFlagBufOffsetListInUse is set in
sl@0
    66
	TSharedChunkBufConfigBase::iFlags. */
sl@0
    67
	TInt iBufferOffsetListStart;
sl@0
    68
	};
sl@0
    69
sl@0
    70
/**
sl@0
    71
The sound driver power handler class.
sl@0
    72
*/
sl@0
    73
class DSoundScPowerHandler : public DPowerHandler
sl@0
    74
	{
sl@0
    75
public:
sl@0
    76
	DSoundScPowerHandler(DSoundScLdd* aChannel);
sl@0
    77
	// Inherited from DPowerHandler
sl@0
    78
	void PowerUp();
sl@0
    79
	void PowerDown(TPowerState aPowerState);
sl@0
    80
private:
sl@0
    81
	DSoundScLdd* iChannel;
sl@0
    82
	};
sl@0
    83
sl@0
    84
/**
sl@0
    85
An object encapsulating an audio data transfer - either record or playback.
sl@0
    86
*/
sl@0
    87
class TSndScTransfer
sl@0
    88
	{
sl@0
    89
public:
sl@0
    90
	enum TTfState
sl@0
    91
		{
sl@0
    92
		/** None of the data for this transfer has been queued on the device. */
sl@0
    93
		ETfNotStarted,
sl@0
    94
		/** Some of the data for this transfer has been queued on the device - but there is more that needs queuing. */
sl@0
    95
		ETfPartlyStarted,
sl@0
    96
		/** All the data for this transfer has been queued on the device - but transfer is not complete. */
sl@0
    97
		ETfFullyStarted,
sl@0
    98
		/** All the data for this transfer has been transferred. */
sl@0
    99
		ETfDone
sl@0
   100
		};
sl@0
   101
public:
sl@0
   102
	TSndScTransfer();
sl@0
   103
	void Init(TUint aId,TInt aChunkOffset,TInt aLength,TAudioBuffer* anAudioBuffer);
sl@0
   104
	inline TInt GetNotStartedLen();
sl@0
   105
	inline TInt GetStartOffset();
sl@0
   106
	inline TInt GetLengthTransferred();
sl@0
   107
	void SetStarted(TInt aLength);
sl@0
   108
	TBool SetCompleted(TInt aLength);
sl@0
   109
public:
sl@0
   110
	/** A value which uniquely identifies a particular audio data transfer. */
sl@0
   111
	TUint iId;
sl@0
   112
	/** The status of this transfer. */
sl@0
   113
	TTfState iTfState;
sl@0
   114
	/** The audio buffer associated with this transfer. */
sl@0
   115
	TAudioBuffer* iAudioBuffer;
sl@0
   116
private:
sl@0
   117
	/** An offset within the shared chunk indicating the progress of the transfer. Data between the initial offset
sl@0
   118
	and this value has either been successfully been transferred or has been queued for transfer. */
sl@0
   119
	TUint iStartedOffset;
sl@0
   120
	/** An offset within the shared chunk indicating the end of the data to be transferred. */
sl@0
   121
	TUint iEndOffset;
sl@0
   122
	/** This holds the count of the number of bytes which have been successfully transferred. */
sl@0
   123
	TInt iLengthTransferred;
sl@0
   124
	/** This holds the count of the number of transfer fragments which are currently in progress for this transfer. */
sl@0
   125
	TInt iTransfersInProgress;
sl@0
   126
	};
sl@0
   127
sl@0
   128
/**
sl@0
   129
An object encapsulating an audio request from the client - either record or playback.
sl@0
   130
*/
sl@0
   131
class TSoundScRequest : public SDblQueLink
sl@0
   132
	{
sl@0
   133
public:
sl@0
   134
	inline TSoundScRequest();
sl@0
   135
	virtual ~TSoundScRequest();
sl@0
   136
	virtual TInt Construct();
sl@0
   137
public:
sl@0
   138
	/** The thread which issued the request and which supplied the request status. */
sl@0
   139
	DThread* iOwningThread;
sl@0
   140
	/** The client request completion object
sl@0
   141
		This is the base class and may be constructed as a derived type*/
sl@0
   142
	TClientRequest* iClientRequest;
sl@0
   143
	};
sl@0
   144
sl@0
   145
/**
sl@0
   146
A play request object.
sl@0
   147
*/
sl@0
   148
class TSoundScPlayRequest : public TSoundScRequest
sl@0
   149
	{
sl@0
   150
public:
sl@0
   151
	TSoundScPlayRequest();
sl@0
   152
	inline void SetFail(TInt aCompletionReason);
sl@0
   153
	inline void UpdateProgress(TInt aLength);
sl@0
   154
	TInt Construct();
sl@0
   155
public:
sl@0
   156
	/** The transfer information associated with this play request. */
sl@0
   157
	TSndScTransfer iTf;
sl@0
   158
	/** The play request flags which were supplied by the client for this request - see KSndFlagLastSample. */
sl@0
   159
	TUint iFlags;
sl@0
   160
	/** The error value to be returned when completing the request. */
sl@0
   161
	TInt iCompletionReason;
sl@0
   162
	};
sl@0
   163
sl@0
   164
/**
sl@0
   165
An object encapsulating a queue of audio requests from the client.
sl@0
   166
*/
sl@0
   167
class TSoundScRequestQueue
sl@0
   168
	{
sl@0
   169
public:
sl@0
   170
	TSoundScRequestQueue(DSoundScLdd* aLdd);
sl@0
   171
	virtual ~TSoundScRequestQueue();
sl@0
   172
	virtual TInt Create();
sl@0
   173
	TSoundScRequest* NextFree();
sl@0
   174
	void Add(TSoundScRequest* aReq);
sl@0
   175
	TSoundScRequest* Remove();
sl@0
   176
	TSoundScRequest* Remove(TSoundScRequest* aReq);
sl@0
   177
	TSoundScRequest* Find(TRequestStatus* aStatus);
sl@0
   178
	void Free(TSoundScRequest* aReq);
sl@0
   179
	inline TBool IsEmpty();
sl@0
   180
	inline TBool IsAnchor(TSoundScRequest* aReq);
sl@0
   181
	void CompleteAll(TInt aCompletionReason,NFastMutex* aMutex=NULL);
sl@0
   182
protected:
sl@0
   183
	/** The queue of pending audio requests. */
sl@0
   184
	SDblQue iPendRequestQ;
sl@0
   185
	/** The queue of unused audio requests. */
sl@0
   186
	SDblQue iUnusedRequestQ;
sl@0
   187
	/** The actual array of request objects. */
sl@0
   188
	TSoundScRequest* iRequest[KMaxSndScRequestsPending];
sl@0
   189
	/** Mutex used to protect the unused queue from corruption */
sl@0
   190
	NFastMutex iUnusedRequestQLock;
sl@0
   191
private:
sl@0
   192
	/** The owning LDD object. */
sl@0
   193
	DSoundScLdd* iLdd;
sl@0
   194
	};
sl@0
   195
sl@0
   196
/**
sl@0
   197
An object encapsulating a queue of play requests from the client.
sl@0
   198
*/
sl@0
   199
class TSoundScPlayRequestQueue : public TSoundScRequestQueue
sl@0
   200
	{
sl@0
   201
public:
sl@0
   202
	TSoundScPlayRequestQueue(DSoundScLdd* aLdd);
sl@0
   203
	virtual TInt Create();
sl@0
   204
	TSoundScPlayRequest* NextRequestForTransfer();
sl@0
   205
	TSoundScPlayRequest* Find(TUint aTransferID,TBool& aIsNextToComplete);
sl@0
   206
	};
sl@0
   207
sl@0
   208
/**
sl@0
   209
The buffer manager base class.
sl@0
   210
*/
sl@0
   211
class DBufferManager : public DBase
sl@0
   212
	{
sl@0
   213
public:
sl@0
   214
	enum TFlushOp
sl@0
   215
		{
sl@0
   216
		/** Flush before a DMA write. */
sl@0
   217
		EFlushBeforeDmaWrite,
sl@0
   218
		/** Flush before a DMA read. */
sl@0
   219
		EFlushBeforeDmaRead,
sl@0
   220
		/** Flush after a DMA read. */
sl@0
   221
		EFlushAfterDmaRead
sl@0
   222
		};
sl@0
   223
public:
sl@0
   224
	DBufferManager(DSoundScLdd* aLdd);
sl@0
   225
	~DBufferManager();
sl@0
   226
	TInt Create(TSoundSharedChunkBufConfig* aBufConfig);
sl@0
   227
	TInt Create(TSoundSharedChunkBufConfig& aBufConfig,TInt aChunkHandle,DThread* anOwningThread);
sl@0
   228
	void FlushData(TInt aChunkOffset,TInt aLength,TFlushOp aFlushOp);
sl@0
   229
protected:
sl@0
   230
	TInt ValidateBufferOffsets(TInt* aBufferOffsetList,TInt aNumBuffers,TInt aBufferSizeInBytes);
sl@0
   231
	TInt ValidateRegion(TUint aChunkOffset,TUint aLength,TAudioBuffer*& anAudioBuffer);
sl@0
   232
	TInt CreateBufferLists(TInt aNumBuffers);
sl@0
   233
	TInt CommitMemoryForBuffer(TInt aChunkOffset,TInt aSize,TBool& aIsContiguous);
sl@0
   234
	void GetChunkCreateInfo(TChunkCreateInfo& aChunkCreateInfo);
sl@0
   235
protected:
sl@0
   236
	/** The owning LDD object. */
sl@0
   237
	DSoundScLdd* iLdd;
sl@0
   238
	/** The chunk which contains the buffers. */
sl@0
   239
	DChunk* iChunk;
sl@0
   240
	/** The linear address in kernel process for the start of the chunk. */
sl@0
   241
	TLinAddr iChunkBase;
sl@0
   242
	/** MMU mapping attributes that the chunk has actually been mapped with. */
sl@0
   243
	TUint32 iChunkMapAttr;
sl@0
   244
	/** The number of buffers. */
sl@0
   245
	TInt iNumBuffers;
sl@0
   246
	/** The actual array of buffer objects. */
sl@0
   247
	TAudioBuffer* iAudioBuffers;
sl@0
   248
	/** The maximum transfer length that the owning audio channel can support in a single data transfer. */
sl@0
   249
	TInt iMaxTransferLen;
sl@0
   250
protected:
sl@0
   251
	friend class DSoundScLdd;
sl@0
   252
	friend class TAudioBuffer;
sl@0
   253
	};
sl@0
   254
sl@0
   255
/**
sl@0
   256
The record buffer manager class.
sl@0
   257
*/
sl@0
   258
class DRecordBufferManager : public DBufferManager
sl@0
   259
	{
sl@0
   260
public:
sl@0
   261
	DRecordBufferManager(DSoundScLdd* aLdd);
sl@0
   262
	void Reset();
sl@0
   263
	inline TAudioBuffer* GetCurrentRecordBuffer();
sl@0
   264
	inline TAudioBuffer* GetNextRecordBuffer();
sl@0
   265
	TAudioBuffer* GetBufferForClient();
sl@0
   266
	TAudioBuffer* SetBufferFilled(TInt aBytesAdded,TInt aTransferResult);
sl@0
   267
	TAudioBuffer* ReleaseBuffer(TInt aChunkOffset);
sl@0
   268
protected:
sl@0
   269
	/** The buffer currently being filled with record data. (Not in any list). */
sl@0
   270
	TAudioBuffer* iCurrentBuffer;
sl@0
   271
	/** The next buffer to use to capture record data. (Not in any list). */
sl@0
   272
	TAudioBuffer* iNextBuffer;
sl@0
   273
	/** A queue of those buffers which are currently free. */
sl@0
   274
	SDblQue iFreeBufferQ;
sl@0
   275
	/** A queue of those buffers which currently contain record data (and which aren't being used by the client). */
sl@0
   276
	SDblQue iCompletedBufferQ;
sl@0
   277
	/** A queue of those buffers which are currently being used by the client. */
sl@0
   278
	SDblQue iInUseBufferQ;
sl@0
   279
	/** A flag set within SetBufferFilled() each time it is necessary to use a buffer from the completed list rather
sl@0
   280
	than the free list as the next buffer for capture (i.e. 'iNextBuffer'). */
sl@0
   281
	TBool iBufOverflow;
sl@0
   282
private:
sl@0
   283
	friend class DSoundScLdd;
sl@0
   284
	};
sl@0
   285
sl@0
   286
/**
sl@0
   287
The class representing a single record/play buffer.
sl@0
   288
*/
sl@0
   289
class TAudioBuffer : public SDblQueLink
sl@0
   290
	{
sl@0
   291
public:
sl@0
   292
	TAudioBuffer();
sl@0
   293
	~TAudioBuffer();
sl@0
   294
	TInt Create(DChunk* aChunk,TInt aChunkOffset,TInt aSize,TBool aIsContiguous,DBufferManager* aBufManager);
sl@0
   295
	TInt GetFragmentLength(TInt aChunkOffset,TInt aLengthRemaining,TPhysAddr& aPhysAddr);
sl@0
   296
	void Flush(DBufferManager::TFlushOp aFlushOp);
sl@0
   297
public:
sl@0
   298
	/** The owning buffer manager. */
sl@0
   299
	DBufferManager* iBufManager;
sl@0
   300
	/** The offset, in bytes, of the start of the buffer within the chunk. */
sl@0
   301
	TInt iChunkOffset;
sl@0
   302
	/** The size of the buffer in bytes. */
sl@0
   303
	TInt iSize;
sl@0
   304
	/** The physical address of the start of the buffer. KPhysAddrInvalid if the buffer is not physically contiguous. */
sl@0
   305
	TPhysAddr iPhysicalAddress;
sl@0
   306
	/** A list of physical addresses for buffer pages. 0 if the buffer is physically contiguous. */
sl@0
   307
	TPhysAddr* iPhysicalPages;
sl@0
   308
	/** Used for record only this is the number of bytes added into this buffer during recording. */
sl@0
   309
	TInt iBytesAdded;
sl@0
   310
	/** Used for record only this is the result of the transfer into this buffer. */
sl@0
   311
	TInt iResult;
sl@0
   312
	};
sl@0
   313
sl@0
   314
/**
sl@0
   315
The physical device driver (PDD) base class for the sound driver - for either playback or record.
sl@0
   316
@publishedPartner
sl@0
   317
@prototype
sl@0
   318
*/
sl@0
   319
class DSoundScPdd : public DBase
sl@0
   320
	{
sl@0
   321
public:
sl@0
   322
	/**
sl@0
   323
	Return the DFC queue to be used by this device.
sl@0
   324
	@param	aUnit	The record or playback unit for which to return the DFC queue.  This is for use by
sl@0
   325
			SMP optimised drivers to return different DFC queues for different units that can
sl@0
   326
			then run on separate CPU cores.
sl@0
   327
	@return The DFC queue to use.
sl@0
   328
	*/
sl@0
   329
	virtual TDfcQue* DfcQ(TInt aUnit)=0;
sl@0
   330
sl@0
   331
	/**
sl@0
   332
	Return the shared chunk create information to be used by this device.
sl@0
   333
	@param aChunkCreateInfo A chunk create info. object to be to be filled with the settings
sl@0
   334
							required for this device.
sl@0
   335
	*/
sl@0
   336
	virtual void GetChunkCreateInfo(TChunkCreateInfo& aChunkCreateInfo)=0;
sl@0
   337
sl@0
   338
	/**
sl@0
   339
	Return the capabilities of this audio device.
sl@0
   340
	This includes information on the direction of the device (i.e. play or record), the number of audio channels
sl@0
   341
	supported (mono, stereo etc), the encoding formats and sample rates supported and so on.
sl@0
   342
	@param aCapsBuf A packaged TSoundFormatsSupportedV02 object to be filled with the capabilities
sl@0
   343
		of this device. This descriptor is in kernel memory and can be accessed directly.
sl@0
   344
	@see TSoundFormatsSupportedV02.
sl@0
   345
	*/
sl@0
   346
	virtual void Caps(TDes8& aCapsBuf) const=0;
sl@0
   347
sl@0
   348
	/**
sl@0
   349
	Return the maximum transfer length in bytes that this device can support in a single data transfer.
sl@0
   350
	(If the device is using the Symbian DMA framework to handle data transfers then the framework handles data
sl@0
   351
	transfers which exceed the maximum transfer length for the platform. However, some PDD implementations
sl@0
   352
	may not use the DMA framework).
sl@0
   353
	@return The maximum transfer length in bytes.
sl@0
   354
	*/
sl@0
   355
	virtual TInt MaxTransferLen() const=0;
sl@0
   356
sl@0
   357
	/**
sl@0
   358
	Configure or reconfigure the device using the configuration supplied.
sl@0
   359
	@param aConfigBuf A packaged TCurrentSoundFormatV02 object which contains the new configuration settings.
sl@0
   360
		This descriptor is in kernel memory and can be accessed directly.
sl@0
   361
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   362
	@see TCurrentSoundFormatV02.
sl@0
   363
	*/
sl@0
   364
	virtual TInt SetConfig(const TDesC8& aConfigBuf)=0;
sl@0
   365
sl@0
   366
	/**
sl@0
   367
	Set the volume or record level.
sl@0
   368
	@param aVolume The play volume or record level to be set - a value in the range 0 to 255. The value 255
sl@0
   369
		equates to the maximum volume and each value below this equates to a 0.5dB step below it.
sl@0
   370
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   371
	*/
sl@0
   372
	virtual TInt SetVolume(TInt aVolume)=0;
sl@0
   373
sl@0
   374
	/**
sl@0
   375
	Prepare the audio device for data transfer.
sl@0
   376
	This may be preparing it either for record or playback - depending on the direction of the channel.
sl@0
   377
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   378
	*/
sl@0
   379
	virtual TInt StartTransfer()=0;
sl@0
   380
sl@0
   381
	/**
sl@0
   382
	Initiate the transfer of a portion of data from/to the audio device.
sl@0
   383
	This may be to either record or playback the data - depending on the direction of the channel. When the transfer is
sl@0
   384
	complete, the PDD should signal this event using the LDD function PlayCallback() or RecordCallback() as appropriate.
sl@0
   385
	@param aTransferID A value assigned by the LDD to allow it to uniquely identify a particular transfer fragment.
sl@0
   386
	@param aLinAddr A linear address within the shared chunk. For playback this is the address of the start of the data
sl@0
   387
		to be transferred. For record, this is the start address for storing the recorded data.
sl@0
   388
	@param aPhysAddr The physical address within the shared chunk that corresponds to the linear address: aLinAddr.
sl@0
   389
	@param aNumBytes The number of bytes to be transferred.
sl@0
   390
	@return KErrNone if the transfer has been initiated successfully;
sl@0
   391
  			KErrNotReady if the device is unable to accept the transfer for the moment;
sl@0
   392
		  	otherwise one of the other system-wide error codes.
sl@0
   393
	*/
sl@0
   394
	virtual TInt TransferData(TUint aTransferID,TLinAddr aLinAddr,TPhysAddr aPhysAddr,TInt aNumBytes)=0;
sl@0
   395
sl@0
   396
	/**
sl@0
   397
	Terminate the transfer of a data to/from the device and to release any resources necessary for transfer.
sl@0
   398
	In the case of playback, this is called soon after the last pending play request from the client has been completed.
sl@0
   399
	In the case of record, the LDD will leave the audio device capturing record data even when there are no record
sl@0
   400
	requests pending from the client. Transfer will only be terminated when the client either issues
sl@0
   401
	RSoundSc::CancelRecordData() or closes the channel. Once this function had been called, the LDD will not issue
sl@0
   402
	any further TransferData() commands without first issueing a StartTransfer() command.
sl@0
   403
	*/
sl@0
   404
	virtual void StopTransfer()=0;
sl@0
   405
sl@0
   406
	/**
sl@0
   407
	Halt the transfer of data to/from the sound device but don't release any resources necessary for transfer.
sl@0
   408
	In the case of playback, if possible, any active transfer should be suspended in such a way that it can be
sl@0
   409
	resumed later - starting from next sample following the one last played.
sl@0
   410
	In the case of record, any active transfer should be aborted. When recording is halted the PDD should signal this event
sl@0
   411
	with a single call of the LDD function RecordCallback() - reporting back any partial data already received. In this
sl@0
   412
	case, if transfer is resumed later, the LDD will issue a new TransferData() request to re-commence data transfer.
sl@0
   413
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   414
	*/
sl@0
   415
	virtual TInt PauseTransfer()=0;
sl@0
   416
sl@0
   417
	/**
sl@0
   418
	Resume the transfer of data to/from the sound device following a request to halt the transfer.
sl@0
   419
	In the case of playback, if possible, any transfer which was active when the device was halted should be resumed -
sl@0
   420
	starting from next sample following the one last played. Once complete, it should be reported using PlayCallback()
sl@0
   421
	as normal.
sl@0
   422
	In the case of record, any active transfer would have been aborted when the device was halted so its just a case
sl@0
   423
	of re-creating the same setup achieved following StartTransfer().
sl@0
   424
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   425
	*/
sl@0
   426
	virtual TInt ResumeTransfer()=0;
sl@0
   427
sl@0
   428
	/**
sl@0
   429
	Power up the sound device. This is called when the channel is first opened and if ever the phone is brought out
sl@0
   430
	of standby mode.
sl@0
   431
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   432
	*/
sl@0
   433
	virtual TInt PowerUp()=0;
sl@0
   434
sl@0
   435
	/**
sl@0
   436
	Power down the sound device. This is called when the channel is closed and just before the phone powers down when
sl@0
   437
	being turned off or going into standby.
sl@0
   438
	*/
sl@0
   439
	virtual void PowerDown()=0;
sl@0
   440
sl@0
   441
	/**
sl@0
   442
	Handle a custom configuration request.
sl@0
   443
	@param aFunction A number identifying the request.
sl@0
   444
	@param aParam A 32-bit value passed to the driver. Its meaning depends on the request.
sl@0
   445
	@return KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   446
	*/
sl@0
   447
	virtual TInt CustomConfig(TInt aFunction,TAny* aParam)=0;
sl@0
   448
sl@0
   449
	/**
sl@0
   450
	Calculates and returns the number of microseconds of data transferred since the start of transfer.
sl@0
   451
	@param aTime	A reference to a variable into which to place the number of microseconds played or recorded.
sl@0
   452
	@param aState	The current state of the transfer (from TSoundScLdd::TState).  Included so that paused
sl@0
   453
					and Active transfers can be treated differently.
sl@0
   454
	@return			KErrNone if successful, otherwise one of the other system wide error codes.
sl@0
   455
	*/
sl@0
   456
	virtual TInt TimeTransferred(TInt64& aTime, TInt aState);
sl@0
   457
protected:
sl@0
   458
	inline DSoundScLdd* Ldd();
sl@0
   459
private:
sl@0
   460
	DSoundScLdd* iLdd;
sl@0
   461
	friend class DSoundScLdd;
sl@0
   462
	};
sl@0
   463
sl@0
   464
/**
sl@0
   465
The logical device (factory class) for the sound driver.
sl@0
   466
*/
sl@0
   467
class DSoundScLddFactory : public DLogicalDevice
sl@0
   468
	{
sl@0
   469
public:
sl@0
   470
	DSoundScLddFactory();
sl@0
   471
	virtual TInt Install();
sl@0
   472
	virtual void GetCaps(TDes8 &aDes) const;
sl@0
   473
	virtual TInt Create(DLogicalChannelBase*& aChannel);
sl@0
   474
	TBool IsUnitOpen(TInt aUnit);
sl@0
   475
	TInt SetUnitOpen(TInt aUnit,TBool aIsOpenSetting);
sl@0
   476
private:
sl@0
   477
	/** Mask to keep track of which units have a channel open on them. */
sl@0
   478
	TUint iUnitsOpenMask;
sl@0
   479
	/** A mutex to protect access to the unit info. mask. */
sl@0
   480
	NFastMutex iUnitInfoMutex;
sl@0
   481
	};
sl@0
   482
sl@0
   483
/**
sl@0
   484
The logical channel class for the sound driver.
sl@0
   485
*/
sl@0
   486
class DSoundScLdd : public DLogicalChannel
sl@0
   487
	{
sl@0
   488
public:
sl@0
   489
	enum TState
sl@0
   490
		{
sl@0
   491
		/** Channel is open - but not configured. */
sl@0
   492
		EOpen,
sl@0
   493
		/** Channel is configured - but inactive. */
sl@0
   494
		EConfigured,
sl@0
   495
		/** Channel is active - recording or playing data. */
sl@0
   496
		EActive,
sl@0
   497
		/** Channel is paused - recording or playing has been suspended. */
sl@0
   498
		EPaused
sl@0
   499
		};
sl@0
   500
public:
sl@0
   501
	DSoundScLdd();
sl@0
   502
	virtual ~DSoundScLdd();
sl@0
   503
	// Inherited from DLogicalChannel
sl@0
   504
	virtual TInt DoCreate(TInt aUnit, const TDesC8* anInfo, const TVersion& aVer);
sl@0
   505
	virtual TInt Request(TInt aReqNo, TAny* a1, TAny* a2);
sl@0
   506
 	virtual TInt SendMsg(TMessageBase* aMsg);
sl@0
   507
	virtual void HandleMsg(TMessageBase* aMsg);
sl@0
   508
	void Shutdown();
sl@0
   509
	// Functions used by the PDD
sl@0
   510
	virtual void PlayCallback(TUint aTransferID,TInt aTransferResult,TInt aBytesPlayed);
sl@0
   511
	virtual void RecordCallback(TUint aTransferID,TInt aTransferResult,TInt aBytesRecorded);
sl@0
   512
	virtual void NotifyChangeOfHwConfigCallback(TBool aHeadsetPresent);
sl@0
   513
	virtual TSoundSharedChunkBufConfig* BufConfig();
sl@0
   514
	virtual TLinAddr ChunkBase();
sl@0
   515
private:
sl@0
   516
	// Implementations for the different kinds of requests
sl@0
   517
	TInt DoControl(TInt aFunction, TAny* a1, TAny* a2,DThread* aThread);
sl@0
   518
	TInt DoRequest(TInt aFunction, TRequestStatus* aStatus, TAny* a1, TAny* a2,DThread* aThread);
sl@0
   519
	TInt DoCancel(TUint aMask);
sl@0
   520
	TInt SetBufferConfig(DThread* aThread);
sl@0
   521
	TInt SetBufferConfig(TInt aChunkHandle,DThread* aThread);
sl@0
   522
	TInt SetSoundConfig();
sl@0
   523
	TInt DoSetSoundConfig(const TCurrentSoundFormatV02& aSoundConfig);
sl@0
   524
	TInt SetVolume(TInt aVolume);
sl@0
   525
	TInt PlayData(TRequestStatus* aStatus,TSoundScPlayRequest* aRequest,DThread* aThread);
sl@0
   526
	TInt RecordData(TRequestStatus* aStatus,TInt* aLengthPtr,DThread* aThread);
sl@0
   527
	TInt StartRecord();
sl@0
   528
	TInt ReleaseBuffer(TInt aChunkOffset);
sl@0
   529
	TInt CustomConfig(TInt aFunction,TAny* aParam);
sl@0
   530
	TInt ValidateConfig(const TCurrentSoundFormatV02& aConfig);
sl@0
   531
	TInt ReAllocBufferConfigInfo(TInt aNumBuffers);
sl@0
   532
	void StartNextPlayTransfers();
sl@0
   533
	inline void CompletePlayRequest(TSoundScPlayRequest* aReq,TInt aResult);
sl@0
   534
	void DoCompletePlayRequest(TSoundScPlayRequest* aReq);
sl@0
   535
	void CompleteAllDonePlayRequests(TSoundScPlayRequest* aReq);
sl@0
   536
	void HandleCurrentRecordBufferDone(TInt aTransferResult);
sl@0
   537
	void CompleteRequest(DThread* aThread,TRequestStatus* aStatus,TInt aReason,TClientRequest* aClientRequest=NULL);
sl@0
   538
	TInt StartNextRecordTransfers();
sl@0
   539
	void StartPlayEofTimer();
sl@0
   540
	void CancelPlayEofTimer();
sl@0
   541
	inline DSoundScPdd* Pdd();
sl@0
   542
	static void PlayEofTimerExpired(TAny* aChannel);
sl@0
   543
	static void PlayEofTimerDfc(TAny* aChannel);
sl@0
   544
	static void PowerUpDfc(TAny* aChannel);
sl@0
   545
	static void PowerDownDfc(TAny* aChannel);
sl@0
   546
	TInt PrePlay(TMessageBase* aMsg);
sl@0
   547
	TInt PreSetBufferChunkCreateOrOpen(TMessageBase* aMsg);
sl@0
   548
	TInt PreSetSoundConfig(TMessageBase* aMsg);
sl@0
   549
private:
sl@0
   550
	/** The handle to the chunk that is returned to the user side code. */
sl@0
   551
	TInt iChunkHandle;
sl@0
   552
	/** The handle of the thread that created the chunk referenced by iChunkHandle */
sl@0
   553
	DThread* iChunkHandleThread;
sl@0
   554
	/** The unit number of this channel. */
sl@0
   555
	TInt iUnit;
sl@0
   556
	/** The data transfer direction for this unit: play or record. */
sl@0
   557
	TSoundDirection iDirection;
sl@0
   558
	/** The operating state of the channel. */
sl@0
   559
	TState iState;
sl@0
   560
	/** Spare. */
sl@0
   561
	TInt iSpare;
sl@0
   562
	/** The buffer manager - managing the shared chunk and the record/play buffers within this. */
sl@0
   563
	DBufferManager* iBufManager;
sl@0
   564
	/** A mutex to protect access to the buffer lists and the pending request list. */
sl@0
   565
	NFastMutex iMutex;
sl@0
   566
	/** The transfer status of the record buffer currently being filled. */
sl@0
   567
	TSndScTransfer iCurrentRecBufTf;
sl@0
   568
	/** The transfer status of the record buffer which is next to be filled. */
sl@0
   569
	TSndScTransfer iNextRecBufTf;
sl@0
   570
	/** The current buffer configuration in the play/record chunk. */
sl@0
   571
	TSoundSharedChunkBufConfig* iBufConfig;
sl@0
   572
	/** The size in bytes of the play/record buffer configuration info. structure. */
sl@0
   573
	TInt iBufConfigSize;
sl@0
   574
	/** The sound driver power handler. */
sl@0
   575
	DSoundScPowerHandler* iPowerHandler;
sl@0
   576
	/** DFC used to handle power down requests from the power manager before a transition into system shutdown/standby. */
sl@0
   577
	TDfc iPowerDownDfc;
sl@0
   578
	/** DFC used to handle power up requests from the power manager following a transition out of system standby. */
sl@0
   579
	TDfc iPowerUpDfc;
sl@0
   580
	/** The capabilities of this device. */
sl@0
   581
	TSoundFormatsSupportedV02 iCaps;
sl@0
   582
	/** The current audio configuration of the driver. */
sl@0
   583
	TCurrentSoundFormatV02 iSoundConfig;
sl@0
   584
	/** The requested audio configuration of the driver before validation. */
sl@0
   585
	TCurrentSoundFormatV02 iTempSoundConfig;
sl@0
   586
	/** A bitmask holding sound configuration status information. */
sl@0
   587
	TUint32 iSoundConfigFlags;
sl@0
   588
	/** The current setting for the record level / play volume. */
sl@0
   589
	TInt iVolume;
sl@0
   590
	/** The queue of pending play or record requests. */
sl@0
   591
	TSoundScRequestQueue* iReqQueue;
sl@0
   592
	/** Used to complete the change of hardware notification. */
sl@0
   593
	TClientDataRequest<TBool>* iNotifyChangeOfHwClientRequest;
sl@0
   594
	/** The thread which has registered for the hardware configuration change notifier. */
sl@0
   595
	DThread* iChangeOfHwConfigThread;
sl@0
   596
	/** A pointer to the headset present boolean variable in client memory which is updated on a notification. */
sl@0
   597
	TBool* iHeadsetPresentStatPtr;
sl@0
   598
	/** To keep track of the number of bytes transferred. */
sl@0
   599
	TInt iBytesTransferred;
sl@0
   600
	/** Count of the number of times the PDD completes a record transfer fragment just after being paused. */
sl@0
   601
	TInt iCompletesWhilePausedCount;
sl@0
   602
	/** A timer used to delay exiting play transfer mode at EOF. */
sl@0
   603
	NTimer iEofTimer;
sl@0
   604
	/** DFC used to delay exiting play transfer mode at EOF. */
sl@0
   605
	TDfc iPlayEofDfc;
sl@0
   606
	/** Used for testing and debugging. */
sl@0
   607
	TUint iTestSettings;
sl@0
   608
	/** A flag to indicate whether the play EOF timer is active. */
sl@0
   609
	TBool iPlayEofTimerActive;
sl@0
   610
	/** Used in debug builds to track that all calls to DThread::Open() are balanced with a close before the driver closes. */
sl@0
   611
	TInt iThreadOpenCount;
sl@0
   612
	/** Used to complete requests in the DFC thread. */
sl@0
   613
	TClientRequest** iClientRequests;
sl@0
   614
sl@0
   615
	friend class DBufferManager;
sl@0
   616
	friend class DSoundScPowerHandler;
sl@0
   617
	friend class TSoundScRequestQueue;
sl@0
   618
	};
sl@0
   619
sl@0
   620
#include <drivers/soundsc.inl>
sl@0
   621
#endif	// __SOUNDSC_H__