os/kernelhwsrv/kernel/eka/include/drivers/dma_v2.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) 2002-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
// include/drivers/dma_v2.h
sl@0
    15
// DMA Framework - Client API v2 definition.
sl@0
    16
//
sl@0
    17
// NB: DMA clients should never include this file directly, but only ever the
sl@0
    18
// generic header file <drivers/dma.h>.
sl@0
    19
//
sl@0
    20
sl@0
    21
#ifndef __DMA_H__
sl@0
    22
#error "dma_v2.h must'n be included directly - use <drivers/dma.h> instead"
sl@0
    23
#endif	// #ifndef __DMA_H__
sl@0
    24
sl@0
    25
#ifndef __DMA_V2_H__
sl@0
    26
#define __DMA_V2_H__
sl@0
    27
sl@0
    28
sl@0
    29
#include <kernel/kernel.h>
sl@0
    30
#include <drivers/dmadefs.h>
sl@0
    31
sl@0
    32
sl@0
    33
//////////////////////////////////////////////////////////////////////////////
sl@0
    34
// Debug Support - KDmaPanicCat is defined in each source file
sl@0
    35
sl@0
    36
#define __DMA_ASSERTD(e) __ASSERT_DEBUG(e, Kern::Fault(KDmaPanicCat, __LINE__))
sl@0
    37
#define __DMA_ASSERTA(e) __ASSERT_ALWAYS(e, Kern::Fault(KDmaPanicCat, __LINE__))
sl@0
    38
#ifdef _DEBUG
sl@0
    39
#define __DMA_CANT_HAPPEN() Kern::Fault(KDmaPanicCat, __LINE__)
sl@0
    40
#define __DMA_DECLARE_INVARIANT public: void Invariant();
sl@0
    41
#define __DMA_INVARIANT() Invariant()
sl@0
    42
#else
sl@0
    43
#define __DMA_CANT_HAPPEN()
sl@0
    44
#define __DMA_DECLARE_INVARIANT
sl@0
    45
#define __DMA_INVARIANT()
sl@0
    46
#endif
sl@0
    47
sl@0
    48
sl@0
    49
//////////////////////////////////////////////////////////////////////////////
sl@0
    50
// INTERFACE EXPOSED TO DEVICE-DRIVERS
sl@0
    51
//////////////////////////////////////////////////////////////////////////////
sl@0
    52
sl@0
    53
//////////////////////////////////////////////////////////////////////////////
sl@0
    54
sl@0
    55
/** Bitmasks used for configuring a DMA request.
sl@0
    56
sl@0
    57
	In general, specify KDmaMemSrc|KDmaIncSrc (resp. KDmaMemDest|KDmaIncDest)
sl@0
    58
	if the source (resp. destination) is a memory buffer and clear
sl@0
    59
	KDmaMemSrc|KDmaIncSrc (resp. KDmaMemDest|KDmaIncDest) if the source
sl@0
    60
	(resp. destination) is a peripheral.
sl@0
    61
sl@0
    62
	If the location is given as a physical address (rather than a linear one)
sl@0
    63
	then also specify KDmaPhysAddrSrc and/or KDmaPhysAddrDest.
sl@0
    64
sl@0
    65
	The EKA1 "Fill Mode" can be implemented by omitting KDmaIncSrc.
sl@0
    66
sl@0
    67
	Some peripherals may require a post-increment address mode.
sl@0
    68
sl@0
    69
	@see DDmaRequest::Fragment
sl@0
    70
	@publishedPartner
sl@0
    71
	@released
sl@0
    72
*/
sl@0
    73
enum TDmaRequestFlags
sl@0
    74
	{
sl@0
    75
	// Note: This enum is only required for backwards compatibility with the
sl@0
    76
	// old DMA framework, it can be removed once this is no longer needed.
sl@0
    77
sl@0
    78
	/** Source is address of memory buffer */
sl@0
    79
	KDmaMemSrc       = 0x01,
sl@0
    80
	/** Destination is address of memory buffer */
sl@0
    81
	KDmaMemDest      = 0x02,
sl@0
    82
	/** Source address must be post-incremented during transfer */
sl@0
    83
	KDmaIncSrc       = 0x04,
sl@0
    84
	/** Destination address must be post-incremented during transfer */
sl@0
    85
	KDmaIncDest      = 0x08,
sl@0
    86
	/** Source address is a physical address (as opposed to a linear one) */
sl@0
    87
	KDmaPhysAddrSrc  = 0x10,
sl@0
    88
	/** Destination address is a physical address (as opposed to a linear one) */
sl@0
    89
	KDmaPhysAddrDest = 0x20,
sl@0
    90
	/** Request a different max transfer size (for instance for test purposes) */
sl@0
    91
	KDmaAltTransferLen = 0x40
sl@0
    92
	};
sl@0
    93
sl@0
    94
sl@0
    95
/** Each hardware or pseudo descriptor is associated with a header.  Headers
sl@0
    96
	are needed because hardware descriptors can not easily be extended to store
sl@0
    97
	additional information.
sl@0
    98
sl@0
    99
	@publishedPartner
sl@0
   100
	@released
sl@0
   101
*/
sl@0
   102
struct SDmaDesHdr
sl@0
   103
	{
sl@0
   104
	SDmaDesHdr* iNext;
sl@0
   105
	};
sl@0
   106
sl@0
   107
/** Pointer to signature of the new extended callback function.
sl@0
   108
sl@0
   109
	TUint - bitmask of one or more TDmaCallbackType values
sl@0
   110
	TDmaResult - just that
sl@0
   111
	TAny* - was provided by client in DDmaRequest constructor
sl@0
   112
	SDmaDesHdr* - points to header (and thus descriptor) which caused a
sl@0
   113
	'descriptor completed' or 'descriptor paused' event.
sl@0
   114
 */
sl@0
   115
typedef void (*TDmaCallback)(TUint, TDmaResult, TAny*, SDmaDesHdr*);
sl@0
   116
sl@0
   117
class TDmaChannel;
sl@0
   118
sl@0
   119
/** A DMA request is a list of fragments small enough to be transferred in one go
sl@0
   120
	by the DMAC.
sl@0
   121
sl@0
   122
	In general, fragmentation is done in the framework by calling Fragment() but
sl@0
   123
	clients with special needs can allocate a blank descriptor list with
sl@0
   124
	ExpandDesList() and customise it to fit their needs.
sl@0
   125
sl@0
   126
	Clients should not set attributes directly, but should use the various functions
sl@0
   127
	instead.
sl@0
   128
sl@0
   129
	This class has not been designed to be called from several concurrent threads.
sl@0
   130
	Multithreaded clients must implement their own locking scheme (via DMutex).
sl@0
   131
sl@0
   132
	Mutexes are used internally to protect data structures accessed both by the
sl@0
   133
	client thread and the DFC thread. Therefore no fast mutex can be held when
sl@0
   134
	calling a request function.
sl@0
   135
sl@0
   136
	@publishedPartner
sl@0
   137
	@released
sl@0
   138
 */
sl@0
   139
class DDmaRequest : public DBase
sl@0
   140
	{
sl@0
   141
	friend class TDmaChannel;
sl@0
   142
sl@0
   143
public:
sl@0
   144
	/** The outcome of the transfer
sl@0
   145
sl@0
   146
		@deprecated
sl@0
   147
		@see TDmaResult
sl@0
   148
	*/
sl@0
   149
	enum TResult {EBadResult=0, EOk, EError};
sl@0
   150
	/** The signature of the completion/failure callback function
sl@0
   151
sl@0
   152
		@deprecated
sl@0
   153
		@see TDmaCallback
sl@0
   154
	*/
sl@0
   155
	typedef void (*TCallback)(TResult, TAny*);
sl@0
   156
sl@0
   157
public:
sl@0
   158
    /** Constructor.
sl@0
   159
sl@0
   160
		Create a new transfer request.
sl@0
   161
sl@0
   162
		@param aChannel The channel this request is bound to.
sl@0
   163
		@param aCb Callback function called on transfer completion or failure
sl@0
   164
		(in channel DFC context).  Can be NULL.
sl@0
   165
		@param aCbArg   Argument passed to callback function.
sl@0
   166
		@param aMaxTransferSize Maximum fragment size.  If not specified, defaults to the maximum size
sl@0
   167
		supported by the DMA controller for the type of transfer that is later scheduled.
sl@0
   168
sl@0
   169
		@deprecated
sl@0
   170
    */
sl@0
   171
	IMPORT_C DDmaRequest(TDmaChannel& aChannel, TCallback aCb=NULL, TAny* aCbArg=NULL,
sl@0
   172
						 TInt aMaxTransferSize=0);
sl@0
   173
sl@0
   174
sl@0
   175
	/** Constructor.
sl@0
   176
sl@0
   177
		Create a new transfer request.
sl@0
   178
sl@0
   179
		@param aChannel The channel this request is bound to.
sl@0
   180
		@param aDmaCb Callback function called on transfer completion or
sl@0
   181
		failure (in channel DFC or ISR context). Can be NULL.
sl@0
   182
		@param aCbArg Argument passed to callback function.
sl@0
   183
		@param aMaxTransferSize Maximum fragment size. If not specified,
sl@0
   184
		defaults to the maximum size supported by the DMA controller for the
sl@0
   185
		type of transfer that is later scheduled.
sl@0
   186
	*/
sl@0
   187
	IMPORT_C DDmaRequest(TDmaChannel& aChannel, TDmaCallback aDmaCb, TAny* aCbArg=NULL,
sl@0
   188
						 TUint aMaxTransferSize=0);
sl@0
   189
sl@0
   190
sl@0
   191
	/** Destructor.
sl@0
   192
sl@0
   193
		Assume the request is not being transferred or pending.
sl@0
   194
    */
sl@0
   195
	IMPORT_C ~DDmaRequest();
sl@0
   196
sl@0
   197
sl@0
   198
	/** Split request into a list of fragments small enough to be fed to the
sl@0
   199
		DMAC.
sl@0
   200
sl@0
   201
		The size of each fragment is smaller than or equal to the maximum
sl@0
   202
		transfer size supported by the DMAC. If the source and/or destination
sl@0
   203
		is memory, each fragment points to memory which is physically
sl@0
   204
		contiguous.
sl@0
   205
sl@0
   206
		The kind of transfer to perform is specified via a set of flags used by
sl@0
   207
		a PIL and a magic cookie passed to the PSL. If the source
sl@0
   208
		(resp. destination) is a peripheral, aSrc (resp. aDest) is treated as a
sl@0
   209
		magic cookie by the PIL and passed straight to the PSL.
sl@0
   210
sl@0
   211
		The request can be uninitialised or may have been fragmented
sl@0
   212
		previously. The previous configuration if any is lost whether or not
sl@0
   213
		the function succeeds.
sl@0
   214
sl@0
   215
		@param aSrc Source memory buffer linear address or peripheral magic
sl@0
   216
		cookie.
sl@0
   217
		@param aDest Destination memory buffer linear address or peripheral
sl@0
   218
		magic cookie.
sl@0
   219
		@param aCount Number of bytes to transfer.
sl@0
   220
		@param aFlags Bitmask characterising the transfer.
sl@0
   221
		@param aPslInfo Hardware-specific information passed to PSL.
sl@0
   222
sl@0
   223
		@return KErrNone if success. KErrArgument if aFlags and/or aPslInfo are
sl@0
   224
		invalid when finding the maximum transfer size. May also fail if
sl@0
   225
		running out of descriptors.
sl@0
   226
sl@0
   227
		@pre The request is not being transferred or pending.
sl@0
   228
		@pre The various parameters must be valid. The PIL or PSL will fault the
sl@0
   229
		kernel if not.
sl@0
   230
sl@0
   231
		@see TDmaRequestFlags
sl@0
   232
sl@0
   233
		@deprecated
sl@0
   234
    */
sl@0
   235
	IMPORT_C TInt Fragment(TUint32 aSrc, TUint32 aDest, TInt aCount, TUint aFlags, TUint32 aPslInfo);
sl@0
   236
sl@0
   237
sl@0
   238
	/** New version of the DMA request fragment function, to be used with the
sl@0
   239
		TDmaTransferArgs structure.
sl@0
   240
	*/
sl@0
   241
	IMPORT_C TInt Fragment(const TDmaTransferArgs& aTransferArgs);
sl@0
   242
sl@0
   243
sl@0
   244
	/** Transfer asynchronously this request.
sl@0
   245
sl@0
   246
		If this request's channel is idle, the request is transferred
sl@0
   247
		immediately. Otherwise, it is queued and transferred later.
sl@0
   248
sl@0
   249
		The client is responsible for ensuring cache consistency before and/or
sl@0
   250
		after the transfer if necessary.
sl@0
   251
sl@0
   252
		@return KErrNone if success, KErrGeneral otherwise.
sl@0
   253
    */
sl@0
   254
	IMPORT_C TInt Queue();
sl@0
   255
sl@0
   256
sl@0
   257
    /** Append new descriptor(s) to existing list.
sl@0
   258
sl@0
   259
		Clients needing to build a custom descriptor list should call this
sl@0
   260
		function to allocate the list and access the resulting list through
sl@0
   261
		iFirstHdr and iLastHdr.
sl@0
   262
sl@0
   263
		Clients should not change the value of iFirstHdr, iLastHdr and the
sl@0
   264
		iNext field of the descriptor headers to ensure descriptors can be
sl@0
   265
		deallocated. Clients are free to change hardware descriptors, including
sl@0
   266
		chaining, in whatever way suit them.
sl@0
   267
sl@0
   268
		Assume the request is not being transferred or pending.
sl@0
   269
sl@0
   270
		@param aCount Number of descriptors to append.
sl@0
   271
sl@0
   272
		@return KErrNone or standard error code.
sl@0
   273
    */
sl@0
   274
	IMPORT_C TInt ExpandDesList(TInt aCount=1);
sl@0
   275
sl@0
   276
sl@0
   277
    /** Append new descriptor(s) to existing list. This function variant
sl@0
   278
		operates on the source port descriptor chain.
sl@0
   279
sl@0
   280
		Works like ExpandDesList except that it uses the iSrcFirstHdr and
sl@0
   281
		iSrcLastHdr fields.
sl@0
   282
sl@0
   283
		@see ExpandDesList
sl@0
   284
sl@0
   285
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   286
		true, otherwise it will just return KErrGeneral.
sl@0
   287
sl@0
   288
		@param aCount Number of descriptors to append.
sl@0
   289
sl@0
   290
		@return KErrNone or standard error code.
sl@0
   291
    */
sl@0
   292
	IMPORT_C TInt ExpandSrcDesList(TInt aCount=1);
sl@0
   293
sl@0
   294
sl@0
   295
    /** Append new descriptor(s) to existing list. This function variant
sl@0
   296
		operates on the destination port descriptor chain.
sl@0
   297
sl@0
   298
		Works like ExpandDesList except that it uses the iDstFirstHdr and
sl@0
   299
		iDstLastHdr fields.
sl@0
   300
sl@0
   301
		@see ExpandDesList
sl@0
   302
sl@0
   303
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   304
		true, otherwise it will just return KErrGeneral.
sl@0
   305
sl@0
   306
		@param aCount Number of descriptors to append.
sl@0
   307
sl@0
   308
		@return KErrNone or standard error code.
sl@0
   309
    */
sl@0
   310
	IMPORT_C TInt ExpandDstDesList(TInt aCount=1);
sl@0
   311
sl@0
   312
sl@0
   313
	/** Free resources associated with this request.
sl@0
   314
sl@0
   315
		Assume the request is not being transferred or pending.
sl@0
   316
    */
sl@0
   317
	IMPORT_C void FreeDesList();
sl@0
   318
sl@0
   319
sl@0
   320
	/** Free resources associated with this request. This function variant
sl@0
   321
		operates on the source port descriptor chain.
sl@0
   322
sl@0
   323
		@see FreeDesList
sl@0
   324
sl@0
   325
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   326
		true, otherwise it will do nothing.
sl@0
   327
    */
sl@0
   328
	IMPORT_C void FreeSrcDesList();
sl@0
   329
sl@0
   330
sl@0
   331
	/** Free resources associated with this request. This function variant
sl@0
   332
		operates on the destination port descriptor chain.
sl@0
   333
sl@0
   334
		@see FreeDesList
sl@0
   335
sl@0
   336
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   337
		true, otherwise it will do nothing.
sl@0
   338
    */
sl@0
   339
	IMPORT_C void FreeDstDesList();
sl@0
   340
sl@0
   341
sl@0
   342
	/** Enables the functionality for counting the transferred source
sl@0
   343
		elements.
sl@0
   344
sl@0
   345
		This function can be called at any time, but the enabled/disabled
sl@0
   346
		status is checked by the framework only at two points in time.
sl@0
   347
sl@0
   348
		The first one is after a request has been queued, and if it is enabled
sl@0
   349
		then the counting will commence as soon as the transfer starts.
sl@0
   350
sl@0
   351
		The second point is when Resume() is called for a paused transfer, and
sl@0
   352
		in this case the following applies. If counting was enabled when the
sl@0
   353
		transfer was paused and it is now disabled then the counting is stopped
sl@0
   354
		at that point and the count value frozen. If counting was disabled when
sl@0
   355
		the transfer was paused and it is now enabled then the counting will
sl@0
   356
		commence when the transfer resumes. (The starting value will depend on
sl@0
   357
		the argument of the enable function.) Otherwise nothing will change,
sl@0
   358
		i.e. counting will either continue normally (enabled/enabled) or
sl@0
   359
		neither stop nor continue (disabled/disabled).
sl@0
   360
sl@0
   361
		Once a status has been set, it remains valid for the entire duration of
sl@0
   362
		the transfer (and beyond, if it is not changed again).
sl@0
   363
sl@0
   364
		@param aResetElementCount If ETrue (the default) then the count
sl@0
   365
		variable will be reset to zero, otherwise it will retain its current
sl@0
   366
		value.
sl@0
   367
sl@0
   368
		@see Queue()
sl@0
   369
		@see TotalNumSrcElementsTransferred()
sl@0
   370
	*/
sl@0
   371
	IMPORT_C void EnableSrcElementCounting(TBool aResetElementCount=ETrue);
sl@0
   372
sl@0
   373
sl@0
   374
	/** Enables the functionality for counting the transferred destination
sl@0
   375
		elements.
sl@0
   376
sl@0
   377
		This function can be called at any time, but the enabled/disabled
sl@0
   378
		status is checked by the framework only at two points in time.
sl@0
   379
sl@0
   380
		The first one is after a request has been queued, and if it is enabled
sl@0
   381
		then the counting will commence as soon as the transfer starts.
sl@0
   382
sl@0
   383
		The second point is when Resume() is called for a paused transfer, and
sl@0
   384
		in this case the following applies. If counting was enabled when the
sl@0
   385
		transfer was paused and it is now disabled then the counting is stopped
sl@0
   386
		at that point and the count value frozen. If counting was disabled when
sl@0
   387
		the transfer was paused and it is now enabled then the counting will
sl@0
   388
		commence when the transfer resumes. (The starting value will depend on
sl@0
   389
		the argument of the enable function.) Otherwise nothing will change,
sl@0
   390
		i.e. counting will either continue normally (enabled/enabled) or
sl@0
   391
		neither stop nor continue (disabled/disabled).
sl@0
   392
sl@0
   393
		Once a status has been set, it remains valid for the entire duration of
sl@0
   394
		the transfer (and beyond, if it is not changed again).
sl@0
   395
sl@0
   396
		@param aResetElementCount If ETrue (the default) then the count
sl@0
   397
		variable will be reset to zero, otherwise it will retain its current
sl@0
   398
		value.
sl@0
   399
sl@0
   400
		@see Queue()
sl@0
   401
		@see TotalNumDstElementsTransferred()
sl@0
   402
	*/
sl@0
   403
	IMPORT_C void EnableDstElementCounting(TBool aResetElementCount=ETrue);
sl@0
   404
sl@0
   405
sl@0
   406
	/** Disables the functionality for counting the transferred source
sl@0
   407
		elements.
sl@0
   408
sl@0
   409
		This function can be called at any time, but the enabled/disabled
sl@0
   410
		status is checked by the framework only at two points in time.
sl@0
   411
sl@0
   412
		The first one is after a request has been queued, and if it is enabled
sl@0
   413
		then the counting will commence as soon as the transfer starts.
sl@0
   414
sl@0
   415
		The second point is when Resume() is called for a paused transfer, and
sl@0
   416
		in this case the following applies. If counting was enabled when the
sl@0
   417
		transfer was paused and it is now disabled then the counting is stopped
sl@0
   418
		at that point and the count value frozen. If counting was disabled when
sl@0
   419
		the transfer was paused and it is now enabled then the counting will
sl@0
   420
		commence when the transfer resumes. (The starting value will depend on
sl@0
   421
		the argument of the enable function.) Otherwise nothing will change,
sl@0
   422
		i.e. counting will either continue normally (enabled/enabled) or
sl@0
   423
		neither stop nor continue (disabled/disabled).
sl@0
   424
sl@0
   425
		Once a status has been set, it remains valid for the entire duration of
sl@0
   426
		the transfer (and beyond, if it is not changed again).
sl@0
   427
sl@0
   428
		@see Queue()
sl@0
   429
		@see TotalNumSrcElementsTransferred()
sl@0
   430
	*/
sl@0
   431
	IMPORT_C void DisableSrcElementCounting();
sl@0
   432
sl@0
   433
sl@0
   434
	/** Disables the functionality for counting the transferred destination
sl@0
   435
		elements.
sl@0
   436
sl@0
   437
		This function can be called at any time, but the enabled/disabled
sl@0
   438
		status is checked by the framework only at two points in time.
sl@0
   439
sl@0
   440
		The first one is after a request has been queued, and if it is enabled
sl@0
   441
		then the counting will commence as soon as the transfer starts.
sl@0
   442
sl@0
   443
		The second point is when Resume() is called for a paused transfer, and
sl@0
   444
		in this case the following applies. If counting was enabled when the
sl@0
   445
		transfer was paused and it is now disabled then the counting is stopped
sl@0
   446
		at that point and the count value frozen. If counting was disabled when
sl@0
   447
		the transfer was paused and it is now enabled then the counting will
sl@0
   448
		commence when the transfer resumes. (The starting value will depend on
sl@0
   449
		the argument of the enable function.) Otherwise nothing will change,
sl@0
   450
		i.e. counting will either continue normally (enabled/enabled) or
sl@0
   451
		neither stop nor continue (disabled/disabled).
sl@0
   452
sl@0
   453
		Once a status has been set, it remains valid for the entire duration of
sl@0
   454
		the transfer (and beyond, if it is not changed again).
sl@0
   455
sl@0
   456
		@see Queue()
sl@0
   457
		@see TotalNumDstElementsTransferred()
sl@0
   458
	*/
sl@0
   459
	IMPORT_C void DisableDstElementCounting();
sl@0
   460
sl@0
   461
sl@0
   462
	/** Returns the number of elements that have been transferred by this
sl@0
   463
		transfer request at the source port.
sl@0
   464
sl@0
   465
		To use this method, the counting functionality has to be explicitly
sl@0
   466
		enabled, either before the transfer request is queued or while it is
sl@0
   467
		paused.
sl@0
   468
sl@0
   469
		@see EnableSrcElementCounting()
sl@0
   470
		@see DisableSrcElementCounting()
sl@0
   471
sl@0
   472
		This function should only be called after the transfer has finished
sl@0
   473
		(completed with or without error, or because it was cancelled) or while
sl@0
   474
		it is paused. Otherwise it may just return 0.
sl@0
   475
sl@0
   476
		@return The number of elements that have been transferred by this
sl@0
   477
		transfer request at the source port.
sl@0
   478
	*/
sl@0
   479
	IMPORT_C TUint32 TotalNumSrcElementsTransferred();
sl@0
   480
sl@0
   481
sl@0
   482
	/** Returns the number of elements that have been transferred by this
sl@0
   483
		transfer request at the destination port.
sl@0
   484
sl@0
   485
		To use this method, the counting functionality has to be explicitly
sl@0
   486
		enabled, either before the transfer request is queued or while it is
sl@0
   487
		paused.
sl@0
   488
sl@0
   489
		@see EnableDstElementCounting()
sl@0
   490
		@see DisableDstElementCounting()
sl@0
   491
sl@0
   492
		This function should only be called after the transfer has finished
sl@0
   493
		(completed with or without error, or because it was cancelled) or while
sl@0
   494
		it is paused. Otherwise it may just return 0.
sl@0
   495
sl@0
   496
		@return The number of elements that have been transferred by this
sl@0
   497
		transfer request at the destination port.
sl@0
   498
	*/
sl@0
   499
	IMPORT_C TUint32 TotalNumDstElementsTransferred();
sl@0
   500
sl@0
   501
sl@0
   502
	/** Returns the number of fragments that this transfer request has been
sl@0
   503
		split into.
sl@0
   504
sl@0
   505
		This number will only be different from 0 once Fragment() has been
sl@0
   506
		called or after descriptors have been manually allocated by the client
sl@0
   507
		using ExpandDesList().
sl@0
   508
sl@0
   509
		If SDmacCaps::iAsymHwDescriptors is true then this function will always
sl@0
   510
		return 0, and SrcFragmentCount() / DstFragmentCount() should be used
sl@0
   511
		instead.
sl@0
   512
sl@0
   513
		@return The number of fragments (descriptors / pseudo descriptors) that
sl@0
   514
		this transfer request has been split into.
sl@0
   515
	 */
sl@0
   516
	IMPORT_C TInt FragmentCount();
sl@0
   517
sl@0
   518
	/** Returns the number of source port fragments that this transfer request
sl@0
   519
		has been split into.
sl@0
   520
sl@0
   521
		This number will only be different from 0 once Fragment() has been
sl@0
   522
		called or after descriptors have been manually allocated by the client
sl@0
   523
		using ExpandSrcDesList().
sl@0
   524
sl@0
   525
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   526
		true, otherwise it will always return 0.
sl@0
   527
sl@0
   528
		@return The number of source port fragments (descriptors) that this
sl@0
   529
		transfer request has been split into.
sl@0
   530
	 */
sl@0
   531
	IMPORT_C TInt SrcFragmentCount();
sl@0
   532
sl@0
   533
sl@0
   534
	/** Returns the number of destination port fragments that this transfer
sl@0
   535
		request has been split into.
sl@0
   536
sl@0
   537
		This number will only be different from 0 once Fragment() has been
sl@0
   538
		called or after descriptors have been manually allocated by the client
sl@0
   539
		using ExpandDstDesList().
sl@0
   540
sl@0
   541
		This function can only be used if SDmacCaps::iAsymHwDescriptors is
sl@0
   542
		true, otherwise it will always return 0.
sl@0
   543
sl@0
   544
		@return The number of destination port fragments (descriptors) that
sl@0
   545
		this transfer request has been split into.
sl@0
   546
	 */
sl@0
   547
	IMPORT_C TInt DstFragmentCount();
sl@0
   548
sl@0
   549
private:
sl@0
   550
	inline void OnDeque();
sl@0
   551
	TUint GetTransferCount(const TDmaTransferArgs& aTransferArgs);
sl@0
   552
	TInt Frag(TDmaTransferArgs& aTransferArgs);
sl@0
   553
	TInt FragSym(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
sl@0
   554
	TInt FragAsym(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
sl@0
   555
	TInt FragAsymSrc(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
sl@0
   556
	TInt FragAsymDst(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
sl@0
   557
	TInt ExpandDesList(TInt aCount, TInt& aDesCount, SDmaDesHdr*& aFirstHdr,
sl@0
   558
					   SDmaDesHdr*& aLastHdr);
sl@0
   559
	void FreeDesList(TInt& aDesCount, SDmaDesHdr*& aFirstHdr, SDmaDesHdr*& aLastHdr);
sl@0
   560
	TInt FragmentCount(const SDmaDesHdr* aHdr);
sl@0
   561
sl@0
   562
public:
sl@0
   563
	// WARNING: The following attributes are accessed both in client and DFC
sl@0
   564
	// context and so accesses must be protected with the channel lock.
sl@0
   565
	TDmaChannel& iChannel;		/**< The channel this request is bound to */
sl@0
   566
	TCallback iCb;			 /**< Called on completion/failure (can be NULL) */
sl@0
   567
	TAny* iCbArg;			 /**< Callback argument */
sl@0
   568
	TDmaCallback iDmaCb;		// the new-style callback function
sl@0
   569
	TAny* iDmaCbArg;			// the new-style callback arg
sl@0
   570
	TBool iIsrCb;				// client wants callback in ISR context
sl@0
   571
	TInt iDesCount;			   /**< The number of fragments in list */
sl@0
   572
	SDmaDesHdr* iFirstHdr;	   /**< The first fragment in the list (or NULL) */
sl@0
   573
	SDmaDesHdr* iLastHdr;	   /**< The last fragment in the list (or NULL) */
sl@0
   574
	TInt iSrcDesCount;		   /**< The number of fragments in list */
sl@0
   575
	SDmaDesHdr* iSrcFirstHdr;  /**< The first fragment in the list (or NULL) */
sl@0
   576
	SDmaDesHdr* iSrcLastHdr;   /**< The last fragment in the list (or NULL) */
sl@0
   577
	TInt iDstDesCount;		   /**< The number of fragments in list */
sl@0
   578
	SDmaDesHdr* iDstFirstHdr;  /**< The first fragment in the list (or NULL) */
sl@0
   579
	SDmaDesHdr* iDstLastHdr;   /**< The last fragment in the list (or NULL) */
sl@0
   580
	SDblQueLink iLink;			/**< The link on channel queue of pending requests */
sl@0
   581
	TBool iQueued;				/**< Indicates whether request is pending or being transferred */
sl@0
   582
	TUint iMaxTransferSize;		/**< Defaults to DMA controller max. transfer size */
sl@0
   583
sl@0
   584
	TUint32 iTotalNumSrcElementsTransferred;
sl@0
   585
	TUint32 iTotalNumDstElementsTransferred;
sl@0
   586
sl@0
   587
	__DMA_DECLARE_INVARIANT
sl@0
   588
	};
sl@0
   589
sl@0
   590
sl@0
   591
//////////////////////////////////////////////////////////////////////////////
sl@0
   592
sl@0
   593
class TDmac;
sl@0
   594
class DmaChannelMgr;
sl@0
   595
class TDmaCancelInfo;
sl@0
   596
sl@0
   597
/** DMA channel base class.
sl@0
   598
sl@0
   599
	Standard derived classes are provided for this channel (see
sl@0
   600
	TDmaSbChannel, TDmaDbChannel, TDmaSgChannel, and TDmaAsymSgChannel).
sl@0
   601
	The base-port implementor will only need to write their own derived
sl@0
   602
	class if one of the standard classes is unsuitable.
sl@0
   603
sl@0
   604
	This class has not been designed to be called from several concurrent
sl@0
   605
	client threads. Multithreaded clients must implement their own locking
sl@0
   606
	scheme (via DMutex).
sl@0
   607
sl@0
   608
	Mutexes are used internally to protect data structures accessed both by the
sl@0
   609
	client thread and the DFC one. Therefore no fast mutex can be held when
sl@0
   610
	calling a channel function.
sl@0
   611
sl@0
   612
	@publishedPartner
sl@0
   613
	@released
sl@0
   614
 */
sl@0
   615
class TDmaChannel
sl@0
   616
	{
sl@0
   617
	friend class DDmaRequest;
sl@0
   618
	friend class TDmac;
sl@0
   619
	friend class DmaChannelMgr;
sl@0
   620
public:
sl@0
   621
sl@0
   622
	/** Information passed by client when opening a channel */
sl@0
   623
	struct SCreateInfo
sl@0
   624
		{
sl@0
   625
		/** Default constructor. Initializes all fields with meaningful default
sl@0
   626
			values.
sl@0
   627
sl@0
   628
			Must be inline (for now) because exporting it would break existing
sl@0
   629
			custom DMA libs as their clients would need the export which would
sl@0
   630
			be missing from the custom .def files.
sl@0
   631
		*/
sl@0
   632
		SCreateInfo() : iPriority(KDmaPriorityNone), iDynChannel(EFalse) {};
sl@0
   633
sl@0
   634
		/** Identifier used by PSL to select channel to open */
sl@0
   635
		TUint32 iCookie;
sl@0
   636
		/** Number of descriptors this channel can use.
sl@0
   637
sl@0
   638
			This number is not used in the upgraded version of the DMA
sl@0
   639
			framework and is kept there only for source compatibility. If the
sl@0
   640
			client is certain that it will only ever use that version, then the
sl@0
   641
			value passed here doesn't matter - the framework will ignore it.
sl@0
   642
sl@0
   643
			@deprecated
sl@0
   644
		 */
sl@0
   645
		TInt iDesCount;
sl@0
   646
		/** DFC queue used to service DMA interrupts.
sl@0
   647
sl@0
   648
			The DFC thread priority must be higher than any client thread
sl@0
   649
			priority to avoid a situation where a transfer completes while
sl@0
   650
			being cancelled and another transfer is started before the DFC
sl@0
   651
			thread gets a chance to run. This would lead to a stray DFC.
sl@0
   652
		*/
sl@0
   653
		TDfcQue* iDfcQ;
sl@0
   654
		/** DFC priority */
sl@0
   655
		TUint8 iDfcPriority;
sl@0
   656
		/** Used by PSL to configure a channel priority (if possible).
sl@0
   657
sl@0
   658
			The default is KDmaPriorityNone (the don't care value).
sl@0
   659
sl@0
   660
		    @see TDmaPriority
sl@0
   661
		*/
sl@0
   662
		TUint iPriority;
sl@0
   663
		/** Request a dynamic DMA channel.
sl@0
   664
sl@0
   665
			If this is set to ETrue then the Open call is for a 'dynamic' as
sl@0
   666
			opposed to a static and solely owned DMA channel. A number of
sl@0
   667
			properties of the opened TDmaChannel object will be different in
sl@0
   668
			that case.
sl@0
   669
sl@0
   670
			The default value is EFalse.
sl@0
   671
		 */
sl@0
   672
		TBool iDynChannel;
sl@0
   673
		};
sl@0
   674
sl@0
   675
public:
sl@0
   676
    /** Opens the DMA channel.
sl@0
   677
sl@0
   678
		Channel selection is done by the hardware-specific layer using a cookie
sl@0
   679
		passed in via aInfo.
sl@0
   680
sl@0
   681
		The client should not delete the returned pointer as the framework owns
sl@0
   682
		channel objects. However, the client should explicitly close the
sl@0
   683
		channel when finished with it.
sl@0
   684
sl@0
   685
		@param aInfo Information passed by caller to select and configure
sl@0
   686
		channel.
sl@0
   687
sl@0
   688
		@param aChannel Points to open channel on successful return. NULL
sl@0
   689
		otherwise.
sl@0
   690
sl@0
   691
		@return KErrNone or standard error code.
sl@0
   692
 	*/
sl@0
   693
	IMPORT_C static TInt Open(const SCreateInfo& aInfo, TDmaChannel*& aChannel);
sl@0
   694
sl@0
   695
sl@0
   696
	/** Closes a previously opened DMA channel.
sl@0
   697
sl@0
   698
		Assumes the channel is idle and all requests have been deleted.
sl@0
   699
sl@0
   700
		The call will cause the resources associated with this channel to be
sl@0
   701
		released, and the pointer/reference to it mustn't therefore be accessed
sl@0
   702
		any longer after the function has returned. The channel pointer should
sl@0
   703
		be set to NULL by the client.
sl@0
   704
 	*/
sl@0
   705
	IMPORT_C void Close();
sl@0
   706
sl@0
   707
sl@0
   708
	/** Logically links this channel to the one specified as an argument, or,
sl@0
   709
		if the argument is NULL, unlinks this channel.
sl@0
   710
sl@0
   711
		The effect of channel linking is that once a transfer on this channel
sl@0
   712
		has finished, instead of causing the associated client callback to be
sl@0
   713
		called, 'aChannel' will be enabled by hardware and a preconfigured
sl@0
   714
		transfer on that channel will start.
sl@0
   715
sl@0
   716
		Note that conceptually 'linking' here always refers to the end of a
sl@0
   717
		channel transfer, not the beginning, i.e. a channel can only be linked
sl@0
   718
		once and always to a successor, never twice or to a predecessor. (This
sl@0
   719
		does not preclude the possibility that two channels are linked in a
sl@0
   720
		circular fashion.)
sl@0
   721
sl@0
   722
		This function can only be used if the DMAC supports logical channel
sl@0
   723
		linking.
sl@0
   724
sl@0
   725
		@see SDmacCaps::iChannelLinking
sl@0
   726
sl@0
   727
		@param aChannel Points to the channel this one should be linked to, or
sl@0
   728
		NULL if this channel is to be unlinked from any other one.
sl@0
   729
sl@0
   730
		@return KErrNone if the channel has been linked or unlinked
sl@0
   731
		successfully, KErrCompletion if this channel was already linked to
sl@0
   732
		aChannel or already unlinked, KErrNotSupported if the DMAC doesn't
sl@0
   733
		support channel linking, KErrArgument if this channel was already
sl@0
   734
		linked to a different channel, KErrGeneral if a general error occurred
sl@0
   735
		preventing a successful outcome.
sl@0
   736
	 */
sl@0
   737
	IMPORT_C TInt LinkToChannel(TDmaChannel* aChannel);
sl@0
   738
sl@0
   739
	/** Pauses an active transfer on this channel.
sl@0
   740
sl@0
   741
		A paused channel transfer can be resumed by calling Resume() or it can
sl@0
   742
		be stopped altogether by calling CancelAll().
sl@0
   743
sl@0
   744
		@see TDmaChannel::Resume()
sl@0
   745
sl@0
   746
		Function can only be used if the DMAC supports this functionality.
sl@0
   747
sl@0
   748
		@see SDmacCaps::iChannelPauseAndResume
sl@0
   749
sl@0
   750
		@return KErrNone if a transfer has been paused successfully,
sl@0
   751
		KErrCompletion if a transfer was already paused, KErrNotSupported if
sl@0
   752
		the DMAC doesn't support channel transfer pausing/resuming, KErrGeneral
sl@0
   753
		if a general error occurred preventing a successful outcome.
sl@0
   754
	 */
sl@0
   755
	IMPORT_C TInt Pause();
sl@0
   756
sl@0
   757
sl@0
   758
	/** Resumes a transfer on this channel that is paused.
sl@0
   759
sl@0
   760
		Resume() can be called to resume channel operation when the transfer is
sl@0
   761
		paused as a result of a previous call to Pause() or because the DMAC
sl@0
   762
		has encountered a Pause bit in a H/W descriptor.
sl@0
   763
sl@0
   764
		@see TDmaChannel::Pause()
sl@0
   765
		@see TDmaCallbackType::EDmaCallbackLinkedListPaused
sl@0
   766
sl@0
   767
		Function can only be used if the DMAC supports this functionality.
sl@0
   768
sl@0
   769
		@see SDmacCaps::iChannelPauseAndResume
sl@0
   770
sl@0
   771
		@return KErrNone if a paused transfer has been resumed successfully,
sl@0
   772
		KErrCompletion if there was no paused transfer, KErrNotSupported if the
sl@0
   773
		DMAC doesn't support channel transfer pausing/resuming, KErrGeneral if
sl@0
   774
		a general error occurred preventing a successful outcome.
sl@0
   775
	 */
sl@0
   776
	IMPORT_C TInt Resume();
sl@0
   777
sl@0
   778
sl@0
   779
	/** Cancels the current request and all the pending ones.
sl@0
   780
	 */
sl@0
   781
	IMPORT_C void CancelAll();
sl@0
   782
sl@0
   783
sl@0
   784
	/** Returns the channel's maximum transfer length based on the passed
sl@0
   785
		arguments.
sl@0
   786
sl@0
   787
		@param aSrcFlags Bitmask characterising transfer source
sl@0
   788
		@see TDmaTransferArgs::iSrcConfig::iFlags
sl@0
   789
sl@0
   790
		@param aDstFlags Bitmask characterising transfer destination
sl@0
   791
		@see TDmaTransferArgs::iDstConfig::iFlags
sl@0
   792
sl@0
   793
		@param aPslInfo Cookie passed to the PSL
sl@0
   794
		@see TDmaTransferArgs::iPslRequestInfo
sl@0
   795
sl@0
   796
		@return 0 if transfer length is not limited, the maximum transfer
sl@0
   797
		length in bytes otherwise.
sl@0
   798
 	*/
sl@0
   799
	IMPORT_C TUint MaxTransferLength(TUint aSrcFlags, TUint aDstFlags, TUint32 aPslInfo);
sl@0
   800
sl@0
   801
sl@0
   802
	/** Retrieves from the PSL the address / memory alignment mask based on the
sl@0
   803
		parameters passed. Some DMA controllers impose alignment constraints on
sl@0
   804
		the base address of memory buffers. This mask is AND'ed against memory
sl@0
   805
		addresses computed during fragmentation.
sl@0
   806
sl@0
   807
		This function needs to be called separately for source and destination.
sl@0
   808
sl@0
   809
		@param aTargetFlags Bitmask characterising transfer source or
sl@0
   810
		destination
sl@0
   811
		@see TDmaTransferArgs::iSrcConfig::iFlags
sl@0
   812
		@see TDmaTransferArgs::iDstConfig::iFlags
sl@0
   813
sl@0
   814
		@param aElementSize Element size used for the transfer. Can be zero if
sl@0
   815
		not known or 'don't care'.
sl@0
   816
sl@0
   817
		@param aPslInfo Cookie passed to the PSL
sl@0
   818
		@see TDmaTransferArgs::iPslRequestInfo
sl@0
   819
sl@0
   820
		@return A value representing the alignment mask (e.g. 3 if buffer must
sl@0
   821
		be 4-byte aligned)
sl@0
   822
	 */
sl@0
   823
	IMPORT_C TUint AddressAlignMask(TUint aTargetFlags, TUint aElementSize,
sl@0
   824
									TUint32 aPslInfo);
sl@0
   825
sl@0
   826
sl@0
   827
	/** Returns a reference to a structure containing the capabilities and
sl@0
   828
		features of the DMA controller associated with this channel.
sl@0
   829
sl@0
   830
		@return A reference to a structure containing the capabilities and
sl@0
   831
		features of the DMA controller associated with this channel.
sl@0
   832
	 */
sl@0
   833
	IMPORT_C const SDmacCaps& DmacCaps();
sl@0
   834
sl@0
   835
sl@0
   836
	/** Sets up once more the transfer request that has just completed, after
sl@0
   837
		optionally having adjusted the transfer parameters as specified.
sl@0
   838
sl@0
   839
		This function is meant to be called exclusively from a client-supplied
sl@0
   840
		callback that is executed in ISR context, and only in response to a
sl@0
   841
		transfer completion notification.
sl@0
   842
sl@0
   843
		If this call returns to the caller with KErrNone then the framework's
sl@0
   844
		ISR handler will subsequently not queue the channel DFC for this
sl@0
   845
		completed request.
sl@0
   846
sl@0
   847
		The parameters specify which changes the framework should apply to the
sl@0
   848
		descriptors of the transfer request before rescheduling it. Arguments
sl@0
   849
		for which no change is required should be passed as their default
sl@0
   850
		values. The parameters correspond to those in the TDmaTransferArgs
sl@0
   851
		struct as follows.
sl@0
   852
sl@0
   853
		@param aSrcAddr @see TDmaTransferArgs::iSrcConfig::iAddr
sl@0
   854
		@param aDstAddr @see TDmaTransferArgs::iDstConfig::iAddr
sl@0
   855
		@param aTransferCount @see TDmaTransferArgs::iTransferCount
sl@0
   856
		@param aPslRequestInfo @see TDmaTransferArgs::iPslRequestInfo
sl@0
   857
		@param aIsrCb If set to ETrue (the default) then the callback of the
sl@0
   858
		rescheduled request will again be called in ISR context
sl@0
   859
sl@0
   860
		Since Epoc::LinearToPhysical() cannot be called in ISR context the
sl@0
   861
		addresses passed into this function must be physical ones, i.e.
sl@0
   862
		TDmaTransferFlags::KDmaPhysAddr is implied.
sl@0
   863
sl@0
   864
		If an address refers to a memory target then
sl@0
   865
		TDmaTransferFlags::KDmaMemIsContiguous is implied as well as no
sl@0
   866
		fragmentation is possible at this point.
sl@0
   867
sl@0
   868
		@pre Must only be called from a 'transfer complete' client callback in
sl@0
   869
		ISR context.
sl@0
   870
sl@0
   871
		@post Framework won't queue the channel DFC for the completed request
sl@0
   872
		in success case.
sl@0
   873
sl@0
   874
		@see DDmaRequest::DDmaRequest(TDmaChannel&, TDmaCallback, TAny*, TUint)
sl@0
   875
		@see TDmaCallbackType::EDmaCallbackRequestCompletion
sl@0
   876
		@see TDmaPILFlags::KDmaRequestCallbackFromIsr
sl@0
   877
sl@0
   878
		@return KErrGeneral if there was an error, KErrNone otherwise.
sl@0
   879
	 */
sl@0
   880
	IMPORT_C TInt IsrRedoRequest(TUint32 aSrcAddr=KPhysAddrInvalid,
sl@0
   881
								 TUint32 aDstAddr=KPhysAddrInvalid,
sl@0
   882
								 TUint aTransferCount=0,
sl@0
   883
								 TUint32 aPslRequestInfo=0,
sl@0
   884
								 TBool aIsrCb=ETrue);
sl@0
   885
sl@0
   886
sl@0
   887
	/** Tests whether the channel is currently opened.
sl@0
   888
sl@0
   889
		@return ETrue if channel is currently opened, EFalse otherwise.
sl@0
   890
sl@0
   891
		NB: This API should not be used any longer.
sl@0
   892
sl@0
   893
		After calling TDmaChannel::Open() successfully the channel is
sl@0
   894
		guaranteed to be open. Therefore there seems no good reason for this
sl@0
   895
		API to exist.
sl@0
   896
sl@0
   897
		@deprecated
sl@0
   898
	 */
sl@0
   899
	inline TBool IsOpened() const;
sl@0
   900
sl@0
   901
sl@0
   902
	/** Tests whether the channel's request queue is currently empty.
sl@0
   903
sl@0
   904
		@return ETrue if request queue is currently empty, EFalse otherwise.
sl@0
   905
	 */
sl@0
   906
	inline TBool IsQueueEmpty() const;
sl@0
   907
sl@0
   908
sl@0
   909
	/** Returns a PSL-specific value which uniquely identifies this channel -
sl@0
   910
		it is used for debug tracing by the PIL.
sl@0
   911
sl@0
   912
		@return PSL-specific value which uniquely identifies this channel.
sl@0
   913
	 */
sl@0
   914
	inline TUint32 PslId() const;
sl@0
   915
sl@0
   916
sl@0
   917
	/** Called by a test harness to force an error when the next fragment is
sl@0
   918
		transferred.
sl@0
   919
sl@0
   920
		@param aFragmentCount The number of consecutive fragments to fail
sl@0
   921
	 */
sl@0
   922
	IMPORT_C TInt FailNext(TInt aFragmentCount);
sl@0
   923
sl@0
   924
sl@0
   925
	/** Called by a test harness to force the DMA controller to miss one or
sl@0
   926
		more interrupts.
sl@0
   927
sl@0
   928
		@param aInterruptCount The number of consecutive interrupts to miss
sl@0
   929
	 */
sl@0
   930
	IMPORT_C TInt MissNextInterrupts(TInt aInterruptCount);
sl@0
   931
sl@0
   932
sl@0
   933
	/** Function allowing platform-specific layer to extend channel API with
sl@0
   934
		new channel-specific operations.
sl@0
   935
sl@0
   936
		@param aCmd Command identifier. Negative values are reserved for use by
sl@0
   937
		Nokia.
sl@0
   938
		@param aArg PSL-specific argument
sl@0
   939
sl@0
   940
		@return KErrNotSupported if aCmd is not supported. PSL-specific value
sl@0
   941
		otherwise.
sl@0
   942
	 */
sl@0
   943
	IMPORT_C TInt Extension(TInt aCmd, TAny* aArg);
sl@0
   944
sl@0
   945
sl@0
   946
	/** This is a function that allows the Platform Specific Layer (PSL) to
sl@0
   947
		extend the DMA API with new channel-independent operations.
sl@0
   948
sl@0
   949
		@param aCmd Command identifier. Negative values are reserved for
sl@0
   950
		Symbian use.
sl@0
   951
		@param aArg PSL-specific.
sl@0
   952
sl@0
   953
		@return KErrNotSupported if aCmd is not supported; a PSL specific value
sl@0
   954
		otherwise.
sl@0
   955
 	*/
sl@0
   956
	IMPORT_C TInt StaticExtension(TInt aCmd, TAny* aArg);
sl@0
   957
sl@0
   958
sl@0
   959
	/** @deprecated
sl@0
   960
		@see DmacCaps()
sl@0
   961
	 */
sl@0
   962
	inline const TDmac* Controller() const;
sl@0
   963
sl@0
   964
	/** @deprecated
sl@0
   965
		@see MaxTransferLength()
sl@0
   966
	 */
sl@0
   967
	inline TInt MaxTransferSize(TUint aFlags, TUint32 aPslInfo);
sl@0
   968
sl@0
   969
	/** @deprecated
sl@0
   970
		@see AddressAlignMask()
sl@0
   971
	 */
sl@0
   972
	inline TUint MemAlignMask(TUint aFlags, TUint32 aPslInfo);
sl@0
   973
sl@0
   974
protected:
sl@0
   975
	// Interface with state machines
sl@0
   976
	TDmaChannel();
sl@0
   977
sl@0
   978
	/** Called by the PIL when adding a new request to the channel's queue.
sl@0
   979
		The implementation should update the channel's state as appropriate
sl@0
   980
		and begin transfer of aReq if possible.
sl@0
   981
sl@0
   982
		@param aReq The request which has been added to the queue
sl@0
   983
	*/
sl@0
   984
	virtual void DoQueue(const DDmaRequest& aReq);
sl@0
   985
sl@0
   986
	/** Called by the PIL in response to a CancelAll call. It should update
sl@0
   987
		the channel state appropriately.
sl@0
   988
	*/
sl@0
   989
	virtual void DoCancelAll() = 0;
sl@0
   990
sl@0
   991
	/** This is called by the PIL when a DDmaRequest is removed from the
sl@0
   992
		channel's queue. In general the implementation simply needs to unlink
sl@0
   993
		the hardware descriptor corresponding to aHdr from the next.
sl@0
   994
sl@0
   995
		Since the PIL links the hardware descriptor chains of adjacent queued
sl@0
   996
		requests (for speed) it is necessary to break the chain when a request
sl@0
   997
		is completed so that the request may be requeued by the client without
sl@0
   998
		having called DDmaRequest::Fragment again.
sl@0
   999
sl@0
  1000
		@param aHdr The header for a descriptor, which must be unlinked
sl@0
  1001
		from its next descriptor (if there is one)
sl@0
  1002
	*/
sl@0
  1003
	virtual void DoUnlink(SDmaDesHdr& aHdr);
sl@0
  1004
sl@0
  1005
	/** Called by the PIL whenever a transfer associated with aCurReq is
sl@0
  1006
		done. The implementation must advance the channel's state and
sl@0
  1007
		may transfer the next header if necessary (the provided
sl@0
  1008
		scatter-gather channel does not do this). It must also report
sl@0
  1009
		back which header was associated with the last transfer to
sl@0
  1010
		complete.
sl@0
  1011
sl@0
  1012
		@param aCurReq The current request.
sl@0
  1013
		@param aCompletedHdr Must be set by the implementation to the header
sl@0
  1014
		of the last transfer to complete.
sl@0
  1015
	*/
sl@0
  1016
	virtual void DoDfc(const DDmaRequest& aCurReq, SDmaDesHdr*& aCompletedHdr);
sl@0
  1017
sl@0
  1018
	/** Called by the PIL whenever a transfer associated with aCurReq is
sl@0
  1019
		done. The implementation must advance the channel's state and
sl@0
  1020
		may start the transfer for the next headers if necessary (the
sl@0
  1021
		provided scatter-gather channels do not do this). If one
sl@0
  1022
		header has a successor but the other is the last in the chain it
sl@0
  1023
		is an error.
sl@0
  1024
sl@0
  1025
		@note Must be implemented by PSL if channel uses asymmetric hardware
sl@0
  1026
		descriptors and is not derived from TDmaAsymSgChannel.
sl@0
  1027
sl@0
  1028
		@param aCurReq The current request.
sl@0
  1029
sl@0
  1030
		@param aSrcCompletedHdr Must be set by the implementation to
sl@0
  1031
		the header of the last source descriptor to complete.
sl@0
  1032
sl@0
  1033
		@param aDstCompletedHdr Must be set by the implementation to
sl@0
  1034
		the header of the last destination descriptor to complete.
sl@0
  1035
	*/
sl@0
  1036
	virtual void DoDfc(const DDmaRequest& aCurReq, SDmaDesHdr*& aSrcCompletedHdr,
sl@0
  1037
					   SDmaDesHdr*& aDstCompletedHdr);
sl@0
  1038
sl@0
  1039
	virtual ~TDmaChannel();
sl@0
  1040
sl@0
  1041
private:
sl@0
  1042
	static void Dfc(TAny*);
sl@0
  1043
	void DoDfc();
sl@0
  1044
	inline void Wait();
sl@0
  1045
	inline void Signal();
sl@0
  1046
	inline void Flash();
sl@0
  1047
	void ResetStateMachine();
sl@0
  1048
sl@0
  1049
protected:
sl@0
  1050
	TDmac* iController;		 // DMAC this channel belongs to (NULL when closed)
sl@0
  1051
	const SDmacCaps* iDmacCaps;	// what is supported by DMAC on this channel
sl@0
  1052
	TUint32 iPslId;			 // unique identifier provided by PSL
sl@0
  1053
	TBool iDynChannel;		 // this is a dynamically allocated channel
sl@0
  1054
	TUint iPriority;		 // hardware priority of this channel
sl@0
  1055
	DMutex* iMutex;			 // for data accessed in both client & DFC context
sl@0
  1056
	SDmaDesHdr* iCurHdr;	 // fragment being transferred or NULL
sl@0
  1057
	SDmaDesHdr** iNullPtr;	 // Pointer to NULL pointer following last fragment
sl@0
  1058
	TDfc iDfc;				  // transfer completion/failure DFC
sl@0
  1059
	TInt iMaxDesCount;		  // maximum number of allocable descriptors
sl@0
  1060
	TInt iAvailDesCount;	  // available number of descriptors
sl@0
  1061
	volatile TUint32 iIsrDfc; // Interface between ISR and DFC:
sl@0
  1062
	enum {KErrorFlagMask = 0x80000000};	   // bit 31 - error flag
sl@0
  1063
	enum {KCancelFlagMask = 0x40000000};   // bit 30 - cancel flag
sl@0
  1064
	enum {KDfcCountMask = 0x3FFFFFFF};	   // bits 0-29 - number of queued DFCs
sl@0
  1065
	SDblQue iReqQ;				 // being/about to be transferred request queue
sl@0
  1066
	TInt iReqCount;				 // number of requests attached to this channel
sl@0
  1067
private:
sl@0
  1068
	TDmaCancelInfo* iCancelInfo; // ...
sl@0
  1069
	TBool iRedoRequest;			 // client ISR callback wants a redo of request
sl@0
  1070
	TBool iIsrCbRequest;		 // request on queue using ISR callback
sl@0
  1071
sl@0
  1072
	__DMA_DECLARE_INVARIANT
sl@0
  1073
	};
sl@0
  1074
sl@0
  1075
sl@0
  1076
//////////////////////////////////////////////////////////////////////////////
sl@0
  1077
// INTERFACE WITH TEST HARNESS
sl@0
  1078
//////////////////////////////////////////////////////////////////////////////
sl@0
  1079
sl@0
  1080
/** Set of information used by test harness.
sl@0
  1081
sl@0
  1082
	@publishedPartner
sl@0
  1083
	@released
sl@0
  1084
*/
sl@0
  1085
struct TDmaTestInfo
sl@0
  1086
	{
sl@0
  1087
	/** Maximum transfer size in bytes for all channels (ie. the minimum of all channels' maximum size)*/
sl@0
  1088
	TUint iMaxTransferSize;
sl@0
  1089
	/** 3->Memory buffers must be 4-byte aligned, 7->8-byte aligned, ... */
sl@0
  1090
	TUint iMemAlignMask;
sl@0
  1091
	/** Cookie to pass to DDmaRequest::Fragment for memory-memory transfer*/
sl@0
  1092
	TUint32 iMemMemPslInfo;
sl@0
  1093
	/** Number of test single-buffer channels */
sl@0
  1094
	TInt iMaxSbChannels;
sl@0
  1095
	/** Pointer to array containing single-buffer test channel ids */
sl@0
  1096
	TUint32* iSbChannels;
sl@0
  1097
	/** Number of test double-buffer channels */
sl@0
  1098
	TInt iMaxDbChannels;
sl@0
  1099
	/** Pointer to array containing double-buffer test channel ids */
sl@0
  1100
	TUint32* iDbChannels;
sl@0
  1101
	/** Number of test scatter-gather channels */
sl@0
  1102
	TInt iMaxSgChannels;
sl@0
  1103
	/** Pointer to array containing scatter-gather test channel ids */
sl@0
  1104
	TUint32* iSgChannels;
sl@0
  1105
	};
sl@0
  1106
sl@0
  1107
sl@0
  1108
/** Provides access to test information structure stored in the PSL.
sl@0
  1109
sl@0
  1110
	Must be implemented by the PSL.
sl@0
  1111
sl@0
  1112
	@publishedPartner
sl@0
  1113
	@released
sl@0
  1114
*/
sl@0
  1115
IMPORT_C const TDmaTestInfo& DmaTestInfo();
sl@0
  1116
sl@0
  1117
/** Provides access to test information structure stored in the PSL.
sl@0
  1118
sl@0
  1119
	Must be implemented by the PSL.
sl@0
  1120
sl@0
  1121
	@publishedPartner
sl@0
  1122
	@released
sl@0
  1123
*/
sl@0
  1124
IMPORT_C const TDmaV2TestInfo& DmaTestInfoV2();
sl@0
  1125
sl@0
  1126
sl@0
  1127
sl@0
  1128
//////////////////////////////////////////////////////////////////////////////
sl@0
  1129
sl@0
  1130
sl@0
  1131
#include <drivers/dma_compat.inl>
sl@0
  1132
#include <drivers/dma_v2.inl>
sl@0
  1133
sl@0
  1134
sl@0
  1135
#endif	// #ifndef __DMA_V2_H__