Update contrib.
1 // Copyright (c) 2002-2009 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0""
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // include/drivers/dma_v2.h
15 // DMA Framework - Client API v2 definition.
17 // NB: DMA clients should never include this file directly, but only ever the
18 // generic header file <drivers/dma.h>.
22 #error "dma_v2.h must'n be included directly - use <drivers/dma.h> instead"
23 #endif // #ifndef __DMA_H__
29 #include <kernel/kernel.h>
30 #include <drivers/dmadefs.h>
33 //////////////////////////////////////////////////////////////////////////////
34 // Debug Support - KDmaPanicCat is defined in each source file
36 #define __DMA_ASSERTD(e) __ASSERT_DEBUG(e, Kern::Fault(KDmaPanicCat, __LINE__))
37 #define __DMA_ASSERTA(e) __ASSERT_ALWAYS(e, Kern::Fault(KDmaPanicCat, __LINE__))
39 #define __DMA_CANT_HAPPEN() Kern::Fault(KDmaPanicCat, __LINE__)
40 #define __DMA_DECLARE_INVARIANT public: void Invariant();
41 #define __DMA_INVARIANT() Invariant()
43 #define __DMA_CANT_HAPPEN()
44 #define __DMA_DECLARE_INVARIANT
45 #define __DMA_INVARIANT()
49 //////////////////////////////////////////////////////////////////////////////
50 // INTERFACE EXPOSED TO DEVICE-DRIVERS
51 //////////////////////////////////////////////////////////////////////////////
53 //////////////////////////////////////////////////////////////////////////////
55 /** Bitmasks used for configuring a DMA request.
57 In general, specify KDmaMemSrc|KDmaIncSrc (resp. KDmaMemDest|KDmaIncDest)
58 if the source (resp. destination) is a memory buffer and clear
59 KDmaMemSrc|KDmaIncSrc (resp. KDmaMemDest|KDmaIncDest) if the source
60 (resp. destination) is a peripheral.
62 If the location is given as a physical address (rather than a linear one)
63 then also specify KDmaPhysAddrSrc and/or KDmaPhysAddrDest.
65 The EKA1 "Fill Mode" can be implemented by omitting KDmaIncSrc.
67 Some peripherals may require a post-increment address mode.
69 @see DDmaRequest::Fragment
75 // Note: This enum is only required for backwards compatibility with the
76 // old DMA framework, it can be removed once this is no longer needed.
78 /** Source is address of memory buffer */
80 /** Destination is address of memory buffer */
82 /** Source address must be post-incremented during transfer */
84 /** Destination address must be post-incremented during transfer */
86 /** Source address is a physical address (as opposed to a linear one) */
87 KDmaPhysAddrSrc = 0x10,
88 /** Destination address is a physical address (as opposed to a linear one) */
89 KDmaPhysAddrDest = 0x20,
90 /** Request a different max transfer size (for instance for test purposes) */
91 KDmaAltTransferLen = 0x40
95 /** Each hardware or pseudo descriptor is associated with a header. Headers
96 are needed because hardware descriptors can not easily be extended to store
97 additional information.
107 /** Pointer to signature of the new extended callback function.
109 TUint - bitmask of one or more TDmaCallbackType values
110 TDmaResult - just that
111 TAny* - was provided by client in DDmaRequest constructor
112 SDmaDesHdr* - points to header (and thus descriptor) which caused a
113 'descriptor completed' or 'descriptor paused' event.
115 typedef void (*TDmaCallback)(TUint, TDmaResult, TAny*, SDmaDesHdr*);
119 /** A DMA request is a list of fragments small enough to be transferred in one go
122 In general, fragmentation is done in the framework by calling Fragment() but
123 clients with special needs can allocate a blank descriptor list with
124 ExpandDesList() and customise it to fit their needs.
126 Clients should not set attributes directly, but should use the various functions
129 This class has not been designed to be called from several concurrent threads.
130 Multithreaded clients must implement their own locking scheme (via DMutex).
132 Mutexes are used internally to protect data structures accessed both by the
133 client thread and the DFC thread. Therefore no fast mutex can be held when
134 calling a request function.
139 class DDmaRequest : public DBase
141 friend class TDmaChannel;
144 /** The outcome of the transfer
149 enum TResult {EBadResult=0, EOk, EError};
150 /** The signature of the completion/failure callback function
155 typedef void (*TCallback)(TResult, TAny*);
160 Create a new transfer request.
162 @param aChannel The channel this request is bound to.
163 @param aCb Callback function called on transfer completion or failure
164 (in channel DFC context). Can be NULL.
165 @param aCbArg Argument passed to callback function.
166 @param aMaxTransferSize Maximum fragment size. If not specified, defaults to the maximum size
167 supported by the DMA controller for the type of transfer that is later scheduled.
171 IMPORT_C DDmaRequest(TDmaChannel& aChannel, TCallback aCb=NULL, TAny* aCbArg=NULL,
172 TInt aMaxTransferSize=0);
177 Create a new transfer request.
179 @param aChannel The channel this request is bound to.
180 @param aDmaCb Callback function called on transfer completion or
181 failure (in channel DFC or ISR context). Can be NULL.
182 @param aCbArg Argument passed to callback function.
183 @param aMaxTransferSize Maximum fragment size. If not specified,
184 defaults to the maximum size supported by the DMA controller for the
185 type of transfer that is later scheduled.
187 IMPORT_C DDmaRequest(TDmaChannel& aChannel, TDmaCallback aDmaCb, TAny* aCbArg=NULL,
188 TUint aMaxTransferSize=0);
193 Assume the request is not being transferred or pending.
195 IMPORT_C ~DDmaRequest();
198 /** Split request into a list of fragments small enough to be fed to the
201 The size of each fragment is smaller than or equal to the maximum
202 transfer size supported by the DMAC. If the source and/or destination
203 is memory, each fragment points to memory which is physically
206 The kind of transfer to perform is specified via a set of flags used by
207 a PIL and a magic cookie passed to the PSL. If the source
208 (resp. destination) is a peripheral, aSrc (resp. aDest) is treated as a
209 magic cookie by the PIL and passed straight to the PSL.
211 The request can be uninitialised or may have been fragmented
212 previously. The previous configuration if any is lost whether or not
213 the function succeeds.
215 @param aSrc Source memory buffer linear address or peripheral magic
217 @param aDest Destination memory buffer linear address or peripheral
219 @param aCount Number of bytes to transfer.
220 @param aFlags Bitmask characterising the transfer.
221 @param aPslInfo Hardware-specific information passed to PSL.
223 @return KErrNone if success. KErrArgument if aFlags and/or aPslInfo are
224 invalid when finding the maximum transfer size. May also fail if
225 running out of descriptors.
227 @pre The request is not being transferred or pending.
228 @pre The various parameters must be valid. The PIL or PSL will fault the
231 @see TDmaRequestFlags
235 IMPORT_C TInt Fragment(TUint32 aSrc, TUint32 aDest, TInt aCount, TUint aFlags, TUint32 aPslInfo);
238 /** New version of the DMA request fragment function, to be used with the
239 TDmaTransferArgs structure.
241 IMPORT_C TInt Fragment(const TDmaTransferArgs& aTransferArgs);
244 /** Transfer asynchronously this request.
246 If this request's channel is idle, the request is transferred
247 immediately. Otherwise, it is queued and transferred later.
249 The client is responsible for ensuring cache consistency before and/or
250 after the transfer if necessary.
252 @return KErrNone if success, KErrGeneral otherwise.
254 IMPORT_C TInt Queue();
257 /** Append new descriptor(s) to existing list.
259 Clients needing to build a custom descriptor list should call this
260 function to allocate the list and access the resulting list through
261 iFirstHdr and iLastHdr.
263 Clients should not change the value of iFirstHdr, iLastHdr and the
264 iNext field of the descriptor headers to ensure descriptors can be
265 deallocated. Clients are free to change hardware descriptors, including
266 chaining, in whatever way suit them.
268 Assume the request is not being transferred or pending.
270 @param aCount Number of descriptors to append.
272 @return KErrNone or standard error code.
274 IMPORT_C TInt ExpandDesList(TInt aCount=1);
277 /** Append new descriptor(s) to existing list. This function variant
278 operates on the source port descriptor chain.
280 Works like ExpandDesList except that it uses the iSrcFirstHdr and
285 This function can only be used if SDmacCaps::iAsymHwDescriptors is
286 true, otherwise it will just return KErrGeneral.
288 @param aCount Number of descriptors to append.
290 @return KErrNone or standard error code.
292 IMPORT_C TInt ExpandSrcDesList(TInt aCount=1);
295 /** Append new descriptor(s) to existing list. This function variant
296 operates on the destination port descriptor chain.
298 Works like ExpandDesList except that it uses the iDstFirstHdr and
303 This function can only be used if SDmacCaps::iAsymHwDescriptors is
304 true, otherwise it will just return KErrGeneral.
306 @param aCount Number of descriptors to append.
308 @return KErrNone or standard error code.
310 IMPORT_C TInt ExpandDstDesList(TInt aCount=1);
313 /** Free resources associated with this request.
315 Assume the request is not being transferred or pending.
317 IMPORT_C void FreeDesList();
320 /** Free resources associated with this request. This function variant
321 operates on the source port descriptor chain.
325 This function can only be used if SDmacCaps::iAsymHwDescriptors is
326 true, otherwise it will do nothing.
328 IMPORT_C void FreeSrcDesList();
331 /** Free resources associated with this request. This function variant
332 operates on the destination port descriptor chain.
336 This function can only be used if SDmacCaps::iAsymHwDescriptors is
337 true, otherwise it will do nothing.
339 IMPORT_C void FreeDstDesList();
342 /** Enables the functionality for counting the transferred source
345 This function can be called at any time, but the enabled/disabled
346 status is checked by the framework only at two points in time.
348 The first one is after a request has been queued, and if it is enabled
349 then the counting will commence as soon as the transfer starts.
351 The second point is when Resume() is called for a paused transfer, and
352 in this case the following applies. If counting was enabled when the
353 transfer was paused and it is now disabled then the counting is stopped
354 at that point and the count value frozen. If counting was disabled when
355 the transfer was paused and it is now enabled then the counting will
356 commence when the transfer resumes. (The starting value will depend on
357 the argument of the enable function.) Otherwise nothing will change,
358 i.e. counting will either continue normally (enabled/enabled) or
359 neither stop nor continue (disabled/disabled).
361 Once a status has been set, it remains valid for the entire duration of
362 the transfer (and beyond, if it is not changed again).
364 @param aResetElementCount If ETrue (the default) then the count
365 variable will be reset to zero, otherwise it will retain its current
369 @see TotalNumSrcElementsTransferred()
371 IMPORT_C void EnableSrcElementCounting(TBool aResetElementCount=ETrue);
374 /** Enables the functionality for counting the transferred destination
377 This function can be called at any time, but the enabled/disabled
378 status is checked by the framework only at two points in time.
380 The first one is after a request has been queued, and if it is enabled
381 then the counting will commence as soon as the transfer starts.
383 The second point is when Resume() is called for a paused transfer, and
384 in this case the following applies. If counting was enabled when the
385 transfer was paused and it is now disabled then the counting is stopped
386 at that point and the count value frozen. If counting was disabled when
387 the transfer was paused and it is now enabled then the counting will
388 commence when the transfer resumes. (The starting value will depend on
389 the argument of the enable function.) Otherwise nothing will change,
390 i.e. counting will either continue normally (enabled/enabled) or
391 neither stop nor continue (disabled/disabled).
393 Once a status has been set, it remains valid for the entire duration of
394 the transfer (and beyond, if it is not changed again).
396 @param aResetElementCount If ETrue (the default) then the count
397 variable will be reset to zero, otherwise it will retain its current
401 @see TotalNumDstElementsTransferred()
403 IMPORT_C void EnableDstElementCounting(TBool aResetElementCount=ETrue);
406 /** Disables the functionality for counting the transferred source
409 This function can be called at any time, but the enabled/disabled
410 status is checked by the framework only at two points in time.
412 The first one is after a request has been queued, and if it is enabled
413 then the counting will commence as soon as the transfer starts.
415 The second point is when Resume() is called for a paused transfer, and
416 in this case the following applies. If counting was enabled when the
417 transfer was paused and it is now disabled then the counting is stopped
418 at that point and the count value frozen. If counting was disabled when
419 the transfer was paused and it is now enabled then the counting will
420 commence when the transfer resumes. (The starting value will depend on
421 the argument of the enable function.) Otherwise nothing will change,
422 i.e. counting will either continue normally (enabled/enabled) or
423 neither stop nor continue (disabled/disabled).
425 Once a status has been set, it remains valid for the entire duration of
426 the transfer (and beyond, if it is not changed again).
429 @see TotalNumSrcElementsTransferred()
431 IMPORT_C void DisableSrcElementCounting();
434 /** Disables the functionality for counting the transferred destination
437 This function can be called at any time, but the enabled/disabled
438 status is checked by the framework only at two points in time.
440 The first one is after a request has been queued, and if it is enabled
441 then the counting will commence as soon as the transfer starts.
443 The second point is when Resume() is called for a paused transfer, and
444 in this case the following applies. If counting was enabled when the
445 transfer was paused and it is now disabled then the counting is stopped
446 at that point and the count value frozen. If counting was disabled when
447 the transfer was paused and it is now enabled then the counting will
448 commence when the transfer resumes. (The starting value will depend on
449 the argument of the enable function.) Otherwise nothing will change,
450 i.e. counting will either continue normally (enabled/enabled) or
451 neither stop nor continue (disabled/disabled).
453 Once a status has been set, it remains valid for the entire duration of
454 the transfer (and beyond, if it is not changed again).
457 @see TotalNumDstElementsTransferred()
459 IMPORT_C void DisableDstElementCounting();
462 /** Returns the number of elements that have been transferred by this
463 transfer request at the source port.
465 To use this method, the counting functionality has to be explicitly
466 enabled, either before the transfer request is queued or while it is
469 @see EnableSrcElementCounting()
470 @see DisableSrcElementCounting()
472 This function should only be called after the transfer has finished
473 (completed with or without error, or because it was cancelled) or while
474 it is paused. Otherwise it may just return 0.
476 @return The number of elements that have been transferred by this
477 transfer request at the source port.
479 IMPORT_C TUint32 TotalNumSrcElementsTransferred();
482 /** Returns the number of elements that have been transferred by this
483 transfer request at the destination port.
485 To use this method, the counting functionality has to be explicitly
486 enabled, either before the transfer request is queued or while it is
489 @see EnableDstElementCounting()
490 @see DisableDstElementCounting()
492 This function should only be called after the transfer has finished
493 (completed with or without error, or because it was cancelled) or while
494 it is paused. Otherwise it may just return 0.
496 @return The number of elements that have been transferred by this
497 transfer request at the destination port.
499 IMPORT_C TUint32 TotalNumDstElementsTransferred();
502 /** Returns the number of fragments that this transfer request has been
505 This number will only be different from 0 once Fragment() has been
506 called or after descriptors have been manually allocated by the client
507 using ExpandDesList().
509 If SDmacCaps::iAsymHwDescriptors is true then this function will always
510 return 0, and SrcFragmentCount() / DstFragmentCount() should be used
513 @return The number of fragments (descriptors / pseudo descriptors) that
514 this transfer request has been split into.
516 IMPORT_C TInt FragmentCount();
518 /** Returns the number of source port fragments that this transfer request
521 This number will only be different from 0 once Fragment() has been
522 called or after descriptors have been manually allocated by the client
523 using ExpandSrcDesList().
525 This function can only be used if SDmacCaps::iAsymHwDescriptors is
526 true, otherwise it will always return 0.
528 @return The number of source port fragments (descriptors) that this
529 transfer request has been split into.
531 IMPORT_C TInt SrcFragmentCount();
534 /** Returns the number of destination port fragments that this transfer
535 request has been split into.
537 This number will only be different from 0 once Fragment() has been
538 called or after descriptors have been manually allocated by the client
539 using ExpandDstDesList().
541 This function can only be used if SDmacCaps::iAsymHwDescriptors is
542 true, otherwise it will always return 0.
544 @return The number of destination port fragments (descriptors) that
545 this transfer request has been split into.
547 IMPORT_C TInt DstFragmentCount();
550 inline void OnDeque();
551 TUint GetTransferCount(const TDmaTransferArgs& aTransferArgs);
552 TInt Frag(TDmaTransferArgs& aTransferArgs);
553 TInt FragSym(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
554 TInt FragAsym(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
555 TInt FragAsymSrc(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
556 TInt FragAsymDst(TDmaTransferArgs& aTransferArgs, TUint aCount, TUint aMaxTransferLen);
557 TInt ExpandDesList(TInt aCount, TInt& aDesCount, SDmaDesHdr*& aFirstHdr,
558 SDmaDesHdr*& aLastHdr);
559 void FreeDesList(TInt& aDesCount, SDmaDesHdr*& aFirstHdr, SDmaDesHdr*& aLastHdr);
560 TInt FragmentCount(const SDmaDesHdr* aHdr);
563 // WARNING: The following attributes are accessed both in client and DFC
564 // context and so accesses must be protected with the channel lock.
565 TDmaChannel& iChannel; /**< The channel this request is bound to */
566 TCallback iCb; /**< Called on completion/failure (can be NULL) */
567 TAny* iCbArg; /**< Callback argument */
568 TDmaCallback iDmaCb; // the new-style callback function
569 TAny* iDmaCbArg; // the new-style callback arg
570 TBool iIsrCb; // client wants callback in ISR context
571 TInt iDesCount; /**< The number of fragments in list */
572 SDmaDesHdr* iFirstHdr; /**< The first fragment in the list (or NULL) */
573 SDmaDesHdr* iLastHdr; /**< The last fragment in the list (or NULL) */
574 TInt iSrcDesCount; /**< The number of fragments in list */
575 SDmaDesHdr* iSrcFirstHdr; /**< The first fragment in the list (or NULL) */
576 SDmaDesHdr* iSrcLastHdr; /**< The last fragment in the list (or NULL) */
577 TInt iDstDesCount; /**< The number of fragments in list */
578 SDmaDesHdr* iDstFirstHdr; /**< The first fragment in the list (or NULL) */
579 SDmaDesHdr* iDstLastHdr; /**< The last fragment in the list (or NULL) */
580 SDblQueLink iLink; /**< The link on channel queue of pending requests */
581 TBool iQueued; /**< Indicates whether request is pending or being transferred */
582 TUint iMaxTransferSize; /**< Defaults to DMA controller max. transfer size */
584 TUint32 iTotalNumSrcElementsTransferred;
585 TUint32 iTotalNumDstElementsTransferred;
587 __DMA_DECLARE_INVARIANT
591 //////////////////////////////////////////////////////////////////////////////
595 class TDmaCancelInfo;
597 /** DMA channel base class.
599 Standard derived classes are provided for this channel (see
600 TDmaSbChannel, TDmaDbChannel, TDmaSgChannel, and TDmaAsymSgChannel).
601 The base-port implementor will only need to write their own derived
602 class if one of the standard classes is unsuitable.
604 This class has not been designed to be called from several concurrent
605 client threads. Multithreaded clients must implement their own locking
608 Mutexes are used internally to protect data structures accessed both by the
609 client thread and the DFC one. Therefore no fast mutex can be held when
610 calling a channel function.
617 friend class DDmaRequest;
619 friend class DmaChannelMgr;
622 /** Information passed by client when opening a channel */
625 /** Default constructor. Initializes all fields with meaningful default
628 Must be inline (for now) because exporting it would break existing
629 custom DMA libs as their clients would need the export which would
630 be missing from the custom .def files.
632 SCreateInfo() : iPriority(KDmaPriorityNone), iDynChannel(EFalse) {};
634 /** Identifier used by PSL to select channel to open */
636 /** Number of descriptors this channel can use.
638 This number is not used in the upgraded version of the DMA
639 framework and is kept there only for source compatibility. If the
640 client is certain that it will only ever use that version, then the
641 value passed here doesn't matter - the framework will ignore it.
646 /** DFC queue used to service DMA interrupts.
648 The DFC thread priority must be higher than any client thread
649 priority to avoid a situation where a transfer completes while
650 being cancelled and another transfer is started before the DFC
651 thread gets a chance to run. This would lead to a stray DFC.
656 /** Used by PSL to configure a channel priority (if possible).
658 The default is KDmaPriorityNone (the don't care value).
663 /** Request a dynamic DMA channel.
665 If this is set to ETrue then the Open call is for a 'dynamic' as
666 opposed to a static and solely owned DMA channel. A number of
667 properties of the opened TDmaChannel object will be different in
670 The default value is EFalse.
676 /** Opens the DMA channel.
678 Channel selection is done by the hardware-specific layer using a cookie
681 The client should not delete the returned pointer as the framework owns
682 channel objects. However, the client should explicitly close the
683 channel when finished with it.
685 @param aInfo Information passed by caller to select and configure
688 @param aChannel Points to open channel on successful return. NULL
691 @return KErrNone or standard error code.
693 IMPORT_C static TInt Open(const SCreateInfo& aInfo, TDmaChannel*& aChannel);
696 /** Closes a previously opened DMA channel.
698 Assumes the channel is idle and all requests have been deleted.
700 The call will cause the resources associated with this channel to be
701 released, and the pointer/reference to it mustn't therefore be accessed
702 any longer after the function has returned. The channel pointer should
703 be set to NULL by the client.
705 IMPORT_C void Close();
708 /** Logically links this channel to the one specified as an argument, or,
709 if the argument is NULL, unlinks this channel.
711 The effect of channel linking is that once a transfer on this channel
712 has finished, instead of causing the associated client callback to be
713 called, 'aChannel' will be enabled by hardware and a preconfigured
714 transfer on that channel will start.
716 Note that conceptually 'linking' here always refers to the end of a
717 channel transfer, not the beginning, i.e. a channel can only be linked
718 once and always to a successor, never twice or to a predecessor. (This
719 does not preclude the possibility that two channels are linked in a
722 This function can only be used if the DMAC supports logical channel
725 @see SDmacCaps::iChannelLinking
727 @param aChannel Points to the channel this one should be linked to, or
728 NULL if this channel is to be unlinked from any other one.
730 @return KErrNone if the channel has been linked or unlinked
731 successfully, KErrCompletion if this channel was already linked to
732 aChannel or already unlinked, KErrNotSupported if the DMAC doesn't
733 support channel linking, KErrArgument if this channel was already
734 linked to a different channel, KErrGeneral if a general error occurred
735 preventing a successful outcome.
737 IMPORT_C TInt LinkToChannel(TDmaChannel* aChannel);
739 /** Pauses an active transfer on this channel.
741 A paused channel transfer can be resumed by calling Resume() or it can
742 be stopped altogether by calling CancelAll().
744 @see TDmaChannel::Resume()
746 Function can only be used if the DMAC supports this functionality.
748 @see SDmacCaps::iChannelPauseAndResume
750 @return KErrNone if a transfer has been paused successfully,
751 KErrCompletion if a transfer was already paused, KErrNotSupported if
752 the DMAC doesn't support channel transfer pausing/resuming, KErrGeneral
753 if a general error occurred preventing a successful outcome.
755 IMPORT_C TInt Pause();
758 /** Resumes a transfer on this channel that is paused.
760 Resume() can be called to resume channel operation when the transfer is
761 paused as a result of a previous call to Pause() or because the DMAC
762 has encountered a Pause bit in a H/W descriptor.
764 @see TDmaChannel::Pause()
765 @see TDmaCallbackType::EDmaCallbackLinkedListPaused
767 Function can only be used if the DMAC supports this functionality.
769 @see SDmacCaps::iChannelPauseAndResume
771 @return KErrNone if a paused transfer has been resumed successfully,
772 KErrCompletion if there was no paused transfer, KErrNotSupported if the
773 DMAC doesn't support channel transfer pausing/resuming, KErrGeneral if
774 a general error occurred preventing a successful outcome.
776 IMPORT_C TInt Resume();
779 /** Cancels the current request and all the pending ones.
781 IMPORT_C void CancelAll();
784 /** Returns the channel's maximum transfer length based on the passed
787 @param aSrcFlags Bitmask characterising transfer source
788 @see TDmaTransferArgs::iSrcConfig::iFlags
790 @param aDstFlags Bitmask characterising transfer destination
791 @see TDmaTransferArgs::iDstConfig::iFlags
793 @param aPslInfo Cookie passed to the PSL
794 @see TDmaTransferArgs::iPslRequestInfo
796 @return 0 if transfer length is not limited, the maximum transfer
797 length in bytes otherwise.
799 IMPORT_C TUint MaxTransferLength(TUint aSrcFlags, TUint aDstFlags, TUint32 aPslInfo);
802 /** Retrieves from the PSL the address / memory alignment mask based on the
803 parameters passed. Some DMA controllers impose alignment constraints on
804 the base address of memory buffers. This mask is AND'ed against memory
805 addresses computed during fragmentation.
807 This function needs to be called separately for source and destination.
809 @param aTargetFlags Bitmask characterising transfer source or
811 @see TDmaTransferArgs::iSrcConfig::iFlags
812 @see TDmaTransferArgs::iDstConfig::iFlags
814 @param aElementSize Element size used for the transfer. Can be zero if
815 not known or 'don't care'.
817 @param aPslInfo Cookie passed to the PSL
818 @see TDmaTransferArgs::iPslRequestInfo
820 @return A value representing the alignment mask (e.g. 3 if buffer must
823 IMPORT_C TUint AddressAlignMask(TUint aTargetFlags, TUint aElementSize,
827 /** Returns a reference to a structure containing the capabilities and
828 features of the DMA controller associated with this channel.
830 @return A reference to a structure containing the capabilities and
831 features of the DMA controller associated with this channel.
833 IMPORT_C const SDmacCaps& DmacCaps();
836 /** Sets up once more the transfer request that has just completed, after
837 optionally having adjusted the transfer parameters as specified.
839 This function is meant to be called exclusively from a client-supplied
840 callback that is executed in ISR context, and only in response to a
841 transfer completion notification.
843 If this call returns to the caller with KErrNone then the framework's
844 ISR handler will subsequently not queue the channel DFC for this
847 The parameters specify which changes the framework should apply to the
848 descriptors of the transfer request before rescheduling it. Arguments
849 for which no change is required should be passed as their default
850 values. The parameters correspond to those in the TDmaTransferArgs
853 @param aSrcAddr @see TDmaTransferArgs::iSrcConfig::iAddr
854 @param aDstAddr @see TDmaTransferArgs::iDstConfig::iAddr
855 @param aTransferCount @see TDmaTransferArgs::iTransferCount
856 @param aPslRequestInfo @see TDmaTransferArgs::iPslRequestInfo
857 @param aIsrCb If set to ETrue (the default) then the callback of the
858 rescheduled request will again be called in ISR context
860 Since Epoc::LinearToPhysical() cannot be called in ISR context the
861 addresses passed into this function must be physical ones, i.e.
862 TDmaTransferFlags::KDmaPhysAddr is implied.
864 If an address refers to a memory target then
865 TDmaTransferFlags::KDmaMemIsContiguous is implied as well as no
866 fragmentation is possible at this point.
868 @pre Must only be called from a 'transfer complete' client callback in
871 @post Framework won't queue the channel DFC for the completed request
874 @see DDmaRequest::DDmaRequest(TDmaChannel&, TDmaCallback, TAny*, TUint)
875 @see TDmaCallbackType::EDmaCallbackRequestCompletion
876 @see TDmaPILFlags::KDmaRequestCallbackFromIsr
878 @return KErrGeneral if there was an error, KErrNone otherwise.
880 IMPORT_C TInt IsrRedoRequest(TUint32 aSrcAddr=KPhysAddrInvalid,
881 TUint32 aDstAddr=KPhysAddrInvalid,
882 TUint aTransferCount=0,
883 TUint32 aPslRequestInfo=0,
887 /** Tests whether the channel is currently opened.
889 @return ETrue if channel is currently opened, EFalse otherwise.
891 NB: This API should not be used any longer.
893 After calling TDmaChannel::Open() successfully the channel is
894 guaranteed to be open. Therefore there seems no good reason for this
899 inline TBool IsOpened() const;
902 /** Tests whether the channel's request queue is currently empty.
904 @return ETrue if request queue is currently empty, EFalse otherwise.
906 inline TBool IsQueueEmpty() const;
909 /** Returns a PSL-specific value which uniquely identifies this channel -
910 it is used for debug tracing by the PIL.
912 @return PSL-specific value which uniquely identifies this channel.
914 inline TUint32 PslId() const;
917 /** Called by a test harness to force an error when the next fragment is
920 @param aFragmentCount The number of consecutive fragments to fail
922 IMPORT_C TInt FailNext(TInt aFragmentCount);
925 /** Called by a test harness to force the DMA controller to miss one or
928 @param aInterruptCount The number of consecutive interrupts to miss
930 IMPORT_C TInt MissNextInterrupts(TInt aInterruptCount);
933 /** Function allowing platform-specific layer to extend channel API with
934 new channel-specific operations.
936 @param aCmd Command identifier. Negative values are reserved for use by
938 @param aArg PSL-specific argument
940 @return KErrNotSupported if aCmd is not supported. PSL-specific value
943 IMPORT_C TInt Extension(TInt aCmd, TAny* aArg);
946 /** This is a function that allows the Platform Specific Layer (PSL) to
947 extend the DMA API with new channel-independent operations.
949 @param aCmd Command identifier. Negative values are reserved for
951 @param aArg PSL-specific.
953 @return KErrNotSupported if aCmd is not supported; a PSL specific value
956 IMPORT_C TInt StaticExtension(TInt aCmd, TAny* aArg);
962 inline const TDmac* Controller() const;
965 @see MaxTransferLength()
967 inline TInt MaxTransferSize(TUint aFlags, TUint32 aPslInfo);
970 @see AddressAlignMask()
972 inline TUint MemAlignMask(TUint aFlags, TUint32 aPslInfo);
975 // Interface with state machines
978 /** Called by the PIL when adding a new request to the channel's queue.
979 The implementation should update the channel's state as appropriate
980 and begin transfer of aReq if possible.
982 @param aReq The request which has been added to the queue
984 virtual void DoQueue(const DDmaRequest& aReq);
986 /** Called by the PIL in response to a CancelAll call. It should update
987 the channel state appropriately.
989 virtual void DoCancelAll() = 0;
991 /** This is called by the PIL when a DDmaRequest is removed from the
992 channel's queue. In general the implementation simply needs to unlink
993 the hardware descriptor corresponding to aHdr from the next.
995 Since the PIL links the hardware descriptor chains of adjacent queued
996 requests (for speed) it is necessary to break the chain when a request
997 is completed so that the request may be requeued by the client without
998 having called DDmaRequest::Fragment again.
1000 @param aHdr The header for a descriptor, which must be unlinked
1001 from its next descriptor (if there is one)
1003 virtual void DoUnlink(SDmaDesHdr& aHdr);
1005 /** Called by the PIL whenever a transfer associated with aCurReq is
1006 done. The implementation must advance the channel's state and
1007 may transfer the next header if necessary (the provided
1008 scatter-gather channel does not do this). It must also report
1009 back which header was associated with the last transfer to
1012 @param aCurReq The current request.
1013 @param aCompletedHdr Must be set by the implementation to the header
1014 of the last transfer to complete.
1016 virtual void DoDfc(const DDmaRequest& aCurReq, SDmaDesHdr*& aCompletedHdr);
1018 /** Called by the PIL whenever a transfer associated with aCurReq is
1019 done. The implementation must advance the channel's state and
1020 may start the transfer for the next headers if necessary (the
1021 provided scatter-gather channels do not do this). If one
1022 header has a successor but the other is the last in the chain it
1025 @note Must be implemented by PSL if channel uses asymmetric hardware
1026 descriptors and is not derived from TDmaAsymSgChannel.
1028 @param aCurReq The current request.
1030 @param aSrcCompletedHdr Must be set by the implementation to
1031 the header of the last source descriptor to complete.
1033 @param aDstCompletedHdr Must be set by the implementation to
1034 the header of the last destination descriptor to complete.
1036 virtual void DoDfc(const DDmaRequest& aCurReq, SDmaDesHdr*& aSrcCompletedHdr,
1037 SDmaDesHdr*& aDstCompletedHdr);
1039 virtual ~TDmaChannel();
1042 static void Dfc(TAny*);
1045 inline void Signal();
1046 inline void Flash();
1047 void ResetStateMachine();
1050 TDmac* iController; // DMAC this channel belongs to (NULL when closed)
1051 const SDmacCaps* iDmacCaps; // what is supported by DMAC on this channel
1052 TUint32 iPslId; // unique identifier provided by PSL
1053 TBool iDynChannel; // this is a dynamically allocated channel
1054 TUint iPriority; // hardware priority of this channel
1055 DMutex* iMutex; // for data accessed in both client & DFC context
1056 SDmaDesHdr* iCurHdr; // fragment being transferred or NULL
1057 SDmaDesHdr** iNullPtr; // Pointer to NULL pointer following last fragment
1058 TDfc iDfc; // transfer completion/failure DFC
1059 TInt iMaxDesCount; // maximum number of allocable descriptors
1060 TInt iAvailDesCount; // available number of descriptors
1061 volatile TUint32 iIsrDfc; // Interface between ISR and DFC:
1062 enum {KErrorFlagMask = 0x80000000}; // bit 31 - error flag
1063 enum {KCancelFlagMask = 0x40000000}; // bit 30 - cancel flag
1064 enum {KDfcCountMask = 0x3FFFFFFF}; // bits 0-29 - number of queued DFCs
1065 SDblQue iReqQ; // being/about to be transferred request queue
1066 TInt iReqCount; // number of requests attached to this channel
1068 TDmaCancelInfo* iCancelInfo; // ...
1069 TBool iRedoRequest; // client ISR callback wants a redo of request
1070 TBool iIsrCbRequest; // request on queue using ISR callback
1072 __DMA_DECLARE_INVARIANT
1076 //////////////////////////////////////////////////////////////////////////////
1077 // INTERFACE WITH TEST HARNESS
1078 //////////////////////////////////////////////////////////////////////////////
1080 /** Set of information used by test harness.
1087 /** Maximum transfer size in bytes for all channels (ie. the minimum of all channels' maximum size)*/
1088 TUint iMaxTransferSize;
1089 /** 3->Memory buffers must be 4-byte aligned, 7->8-byte aligned, ... */
1090 TUint iMemAlignMask;
1091 /** Cookie to pass to DDmaRequest::Fragment for memory-memory transfer*/
1092 TUint32 iMemMemPslInfo;
1093 /** Number of test single-buffer channels */
1094 TInt iMaxSbChannels;
1095 /** Pointer to array containing single-buffer test channel ids */
1096 TUint32* iSbChannels;
1097 /** Number of test double-buffer channels */
1098 TInt iMaxDbChannels;
1099 /** Pointer to array containing double-buffer test channel ids */
1100 TUint32* iDbChannels;
1101 /** Number of test scatter-gather channels */
1102 TInt iMaxSgChannels;
1103 /** Pointer to array containing scatter-gather test channel ids */
1104 TUint32* iSgChannels;
1108 /** Provides access to test information structure stored in the PSL.
1110 Must be implemented by the PSL.
1115 IMPORT_C const TDmaTestInfo& DmaTestInfo();
1117 /** Provides access to test information structure stored in the PSL.
1119 Must be implemented by the PSL.
1124 IMPORT_C const TDmaV2TestInfo& DmaTestInfoV2();
1128 //////////////////////////////////////////////////////////////////////////////
1131 #include <drivers/dma_compat.inl>
1132 #include <drivers/dma_v2.inl>
1135 #endif // #ifndef __DMA_V2_H__