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__
|