xref: /openbmc/u-boot/drivers/dma/MCD_dmaApi.c (revision 48038c4acb58457210f3432f3d6f191c5bb1a9ce)
1 /*
2  * Copyright (C) 2004-2007 Freescale Semiconductor, Inc.
3  *
4  * SPDX-License-Identifier:	GPL-2.0+
5  */
6 
7 /*Main C file for multi-channel DMA API. */
8 
9 #include <common.h>
10 
11 #include <MCD_dma.h>
12 #include <MCD_tasksInit.h>
13 #include <MCD_progCheck.h>
14 
15 /********************************************************************/
16 /* This is an API-internal pointer to the DMA's registers */
17 dmaRegs *MCD_dmaBar;
18 
19 /*
20  * These are the real and model task tables as generated by the
21  * build process
22  */
23 extern TaskTableEntry MCD_realTaskTableSrc[NCHANNELS];
24 extern TaskTableEntry MCD_modelTaskTableSrc[NUMOFVARIANTS];
25 
26 /*
27  * However, this (usually) gets relocated to on-chip SRAM, at which
28  * point we access them as these tables
29  */
30 volatile TaskTableEntry *MCD_taskTable;
31 TaskTableEntry *MCD_modelTaskTable;
32 
33 /*
34  * MCD_chStatus[] is an array of status indicators for remembering
35  * whether a DMA has ever been attempted on each channel, pausing
36  * status, etc.
37  */
38 static int MCD_chStatus[NCHANNELS] = {
39 	MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA,
40 	MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA,
41 	MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA,
42 	MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA, MCD_NO_DMA
43 };
44 
45 /* Prototypes for local functions */
46 static void MCD_memcpy(int *dest, int *src, u32 size);
47 static void MCD_resmActions(int channel);
48 
49 /*
50  * Buffer descriptors used for storage of progress info for single Dmas
51  * Also used as storage for the DMA for CRCs for single DMAs
52  * Otherwise, the DMA does not parse these buffer descriptors
53  */
54 #ifdef MCD_INCLUDE_EU
55 extern MCD_bufDesc MCD_singleBufDescs[NCHANNELS];
56 #else
57 MCD_bufDesc MCD_singleBufDescs[NCHANNELS];
58 #endif
59 MCD_bufDesc *MCD_relocBuffDesc;
60 
61 /* Defines for the debug control register's functions */
62 #define DBG_CTL_COMP1_TASK	(0x00002000)
63 #define DBG_CTL_ENABLE		(DBG_CTL_AUTO_ARM	| \
64 				 DBG_CTL_BREAK		| \
65 				 DBG_CTL_INT_BREAK	| \
66 				 DBG_CTL_COMP1_TASK)
67 #define DBG_CTL_DISABLE		(DBG_CTL_AUTO_ARM	| \
68 				 DBG_CTL_INT_BREAK	| \
69 				 DBG_CTL_COMP1_TASK)
70 #define DBG_KILL_ALL_STAT	(0xFFFFFFFF)
71 
72 /* Offset to context save area where progress info is stored */
73 #define CSAVE_OFFSET		10
74 
75 /* Defines for Byte Swapping */
76 #define MCD_BYTE_SWAP_KILLER	0xFFF8888F
77 #define MCD_NO_BYTE_SWAP_ATALL	0x00040000
78 
79 /* Execution Unit Identifiers */
80 #define MAC			0	/* legacy - not used */
81 #define LUAC			1	/* legacy - not used */
82 #define CRC			2	/* legacy - not used */
83 #define LURC			3	/* Logic Unit with CRC */
84 
85 /* Task Identifiers */
86 #define TASK_CHAINNOEU		0
87 #define TASK_SINGLENOEU		1
88 #ifdef MCD_INCLUDE_EU
89 #define TASK_CHAINEU		2
90 #define TASK_SINGLEEU		3
91 #define TASK_FECRX		4
92 #define TASK_FECTX		5
93 #else
94 #define TASK_CHAINEU		0
95 #define TASK_SINGLEEU		1
96 #define TASK_FECRX		2
97 #define TASK_FECTX		3
98 #endif
99 
100 /*
101  * Structure to remember which variant is on which channel
102  * TBD- need this?
103  */
104 typedef struct MCD_remVariants_struct MCD_remVariant;
105 struct MCD_remVariants_struct {
106 	int remDestRsdIncr[NCHANNELS];	/* -1,0,1 */
107 	int remSrcRsdIncr[NCHANNELS];	/* -1,0,1 */
108 	s16 remDestIncr[NCHANNELS];	/* DestIncr */
109 	s16 remSrcIncr[NCHANNELS];	/* srcIncr */
110 	u32 remXferSize[NCHANNELS];	/* xferSize */
111 };
112 
113 /* Structure to remember the startDma parameters for each channel */
114 MCD_remVariant MCD_remVariants;
115 /********************************************************************/
116 /* Function: MCD_initDma
117  * Purpose:  Initializes the DMA API by setting up a pointer to the DMA
118  *           registers, relocating and creating the appropriate task
119  *           structures, and setting up some global settings
120  * Arguments:
121  *  dmaBarAddr    - pointer to the multichannel DMA registers
122  *  taskTableDest - location to move DMA task code and structs to
123  *  flags         - operational parameters
124  * Return Value:
125  *  MCD_TABLE_UNALIGNED if taskTableDest is not 512-byte aligned
126  *  MCD_OK otherwise
127  */
128 extern u32 MCD_funcDescTab0[];
129 
130 int MCD_initDma(dmaRegs * dmaBarAddr, void *taskTableDest, u32 flags)
131 {
132 	int i;
133 	TaskTableEntry *entryPtr;
134 
135 	/* setup the local pointer to register set */
136 	MCD_dmaBar = dmaBarAddr;
137 
138 	/* do we need to move/create a task table */
139 	if ((flags & MCD_RELOC_TASKS) != 0) {
140 		int fixedSize;
141 		u32 *fixedPtr;
142 		/*int *tablePtr = taskTableDest;TBD */
143 		int varTabsOffset, funcDescTabsOffset, contextSavesOffset;
144 		int taskDescTabsOffset;
145 		int taskTableSize, varTabsSize, funcDescTabsSize,
146 		    contextSavesSize;
147 		int taskDescTabSize;
148 
149 		int i;
150 
151 		/* check if physical address is aligned on 512 byte boundary */
152 		if (((u32) taskTableDest & 0x000001ff) != 0)
153 			return (MCD_TABLE_UNALIGNED);
154 
155 		/* set up local pointer to task Table */
156 		MCD_taskTable = taskTableDest;
157 
158 		/*
159 		 * Create a task table:
160 		 * - compute aligned base offsets for variable tables and
161 		 *   function descriptor tables, then
162 		 * - loop through the task table and setup the pointers
163 		 * - copy over model task table with the the actual task
164 		 *   descriptor tables
165 		 */
166 
167 		taskTableSize = NCHANNELS * sizeof(TaskTableEntry);
168 		/* align variable tables to size */
169 		varTabsOffset = taskTableSize + (u32) taskTableDest;
170 		if ((varTabsOffset & (VAR_TAB_SIZE - 1)) != 0)
171 			varTabsOffset =
172 			    (varTabsOffset + VAR_TAB_SIZE) & (~VAR_TAB_SIZE);
173 		/* align function descriptor tables */
174 		varTabsSize = NCHANNELS * VAR_TAB_SIZE;
175 		funcDescTabsOffset = varTabsOffset + varTabsSize;
176 
177 		if ((funcDescTabsOffset & (FUNCDESC_TAB_SIZE - 1)) != 0)
178 			funcDescTabsOffset =
179 			    (funcDescTabsOffset +
180 			     FUNCDESC_TAB_SIZE) & (~FUNCDESC_TAB_SIZE);
181 
182 		funcDescTabsSize = FUNCDESC_TAB_NUM * FUNCDESC_TAB_SIZE;
183 		contextSavesOffset = funcDescTabsOffset + funcDescTabsSize;
184 		contextSavesSize = (NCHANNELS * CONTEXT_SAVE_SIZE);
185 		fixedSize =
186 		    taskTableSize + varTabsSize + funcDescTabsSize +
187 		    contextSavesSize;
188 
189 		/* zero the thing out */
190 		fixedPtr = (u32 *) taskTableDest;
191 		for (i = 0; i < (fixedSize / 4); i++)
192 			fixedPtr[i] = 0;
193 
194 		entryPtr = (TaskTableEntry *) MCD_taskTable;
195 		/* set up fixed pointers */
196 		for (i = 0; i < NCHANNELS; i++) {
197 			/* update ptr to local value */
198 			entryPtr[i].varTab = (u32) varTabsOffset;
199 			entryPtr[i].FDTandFlags =
200 			    (u32) funcDescTabsOffset | MCD_TT_FLAGS_DEF;
201 			entryPtr[i].contextSaveSpace = (u32) contextSavesOffset;
202 			varTabsOffset += VAR_TAB_SIZE;
203 #ifdef MCD_INCLUDE_EU
204 			/* if not there is only one, just point to the
205 			   same one */
206 			funcDescTabsOffset += FUNCDESC_TAB_SIZE;
207 #endif
208 			contextSavesOffset += CONTEXT_SAVE_SIZE;
209 		}
210 		/* copy over the function descriptor table */
211 		for (i = 0; i < FUNCDESC_TAB_NUM; i++) {
212 			MCD_memcpy((void *)(entryPtr[i].
213 					    FDTandFlags & ~MCD_TT_FLAGS_MASK),
214 				   (void *)MCD_funcDescTab0, FUNCDESC_TAB_SIZE);
215 		}
216 
217 		/* copy model task table to where the context saves stuff
218 		   leaves off */
219 		MCD_modelTaskTable = (TaskTableEntry *) contextSavesOffset;
220 
221 		MCD_memcpy((void *)MCD_modelTaskTable,
222 			   (void *)MCD_modelTaskTableSrc,
223 			   NUMOFVARIANTS * sizeof(TaskTableEntry));
224 
225 		/* point to local version of model task table */
226 		entryPtr = MCD_modelTaskTable;
227 		taskDescTabsOffset = (u32) MCD_modelTaskTable +
228 		    (NUMOFVARIANTS * sizeof(TaskTableEntry));
229 
230 		/* copy actual task code and update TDT ptrs in local
231 		   model task table */
232 		for (i = 0; i < NUMOFVARIANTS; i++) {
233 			taskDescTabSize =
234 			    entryPtr[i].TDTend - entryPtr[i].TDTstart + 4;
235 			MCD_memcpy((void *)taskDescTabsOffset,
236 				   (void *)entryPtr[i].TDTstart,
237 				   taskDescTabSize);
238 			entryPtr[i].TDTstart = (u32) taskDescTabsOffset;
239 			taskDescTabsOffset += taskDescTabSize;
240 			entryPtr[i].TDTend = (u32) taskDescTabsOffset - 4;
241 		}
242 #ifdef MCD_INCLUDE_EU
243 		/* Tack single DMA BDs onto end of code so API controls
244 		   where they are since DMA might write to them */
245 		MCD_relocBuffDesc =
246 		    (MCD_bufDesc *) (entryPtr[NUMOFVARIANTS - 1].TDTend + 4);
247 #else
248 		/* DMA does not touch them so they can be wherever and we
249 		   don't need to waste SRAM on them */
250 		MCD_relocBuffDesc = MCD_singleBufDescs;
251 #endif
252 	} else {
253 		/* point the would-be relocated task tables and the
254 		   buffer descriptors to the ones the linker generated */
255 
256 		if (((u32) MCD_realTaskTableSrc & 0x000001ff) != 0)
257 			return (MCD_TABLE_UNALIGNED);
258 
259 		/* need to add code to make sure that every thing else is
260 		   aligned properly TBD. this is problematic if we init
261 		   more than once or after running tasks, need to add
262 		   variable to see if we have aleady init'd */
263 		entryPtr = MCD_realTaskTableSrc;
264 		for (i = 0; i < NCHANNELS; i++) {
265 			if (((entryPtr[i].varTab & (VAR_TAB_SIZE - 1)) != 0) ||
266 			    ((entryPtr[i].
267 			      FDTandFlags & (FUNCDESC_TAB_SIZE - 1)) != 0))
268 				return (MCD_TABLE_UNALIGNED);
269 		}
270 
271 		MCD_taskTable = MCD_realTaskTableSrc;
272 		MCD_modelTaskTable = MCD_modelTaskTableSrc;
273 		MCD_relocBuffDesc = MCD_singleBufDescs;
274 	}
275 
276 	/* Make all channels as totally inactive, and remember them as such: */
277 
278 	MCD_dmaBar->taskbar = (u32) MCD_taskTable;
279 	for (i = 0; i < NCHANNELS; i++) {
280 		MCD_dmaBar->taskControl[i] = 0x0;
281 		MCD_chStatus[i] = MCD_NO_DMA;
282 	}
283 
284 	/* Set up pausing mechanism to inactive state: */
285 	/* no particular values yet for either comparator registers */
286 	MCD_dmaBar->debugComp1 = 0;
287 	MCD_dmaBar->debugComp2 = 0;
288 	MCD_dmaBar->debugControl = DBG_CTL_DISABLE;
289 	MCD_dmaBar->debugStatus = DBG_KILL_ALL_STAT;
290 
291 	/* enable or disable commbus prefetch, really need an ifdef or
292 	   something to keep from trying to set this in the 8220 */
293 	if ((flags & MCD_COMM_PREFETCH_EN) != 0)
294 		MCD_dmaBar->ptdControl &= ~PTD_CTL_COMM_PREFETCH;
295 	else
296 		MCD_dmaBar->ptdControl |= PTD_CTL_COMM_PREFETCH;
297 
298 	return (MCD_OK);
299 }
300 
301 /*********************** End of MCD_initDma() ***********************/
302 
303 /********************************************************************/
304 /* Function:   MCD_dmaStatus
305  * Purpose:    Returns the status of the DMA on the requested channel
306  * Arguments:  channel - channel number
307  * Returns:    Predefined status indicators
308  */
309 int MCD_dmaStatus(int channel)
310 {
311 	u16 tcrValue;
312 
313 	if ((channel < 0) || (channel >= NCHANNELS))
314 		return (MCD_CHANNEL_INVALID);
315 
316 	tcrValue = MCD_dmaBar->taskControl[channel];
317 	if ((tcrValue & TASK_CTL_EN) == 0) {	/* nothing running */
318 		/* if last reported with task enabled */
319 		if (MCD_chStatus[channel] == MCD_RUNNING
320 		    || MCD_chStatus[channel] == MCD_IDLE)
321 			MCD_chStatus[channel] = MCD_DONE;
322 	} else {		/* something is running */
323 
324 		/* There are three possibilities: paused, running or idle. */
325 		if (MCD_chStatus[channel] == MCD_RUNNING
326 		    || MCD_chStatus[channel] == MCD_IDLE) {
327 			MCD_dmaBar->ptdDebug = PTD_DBG_TSK_VLD_INIT;
328 			/* This register is selected to know which initiator is
329 			   actually asserted. */
330 			if ((MCD_dmaBar->ptdDebug >> channel) & 0x1)
331 				MCD_chStatus[channel] = MCD_RUNNING;
332 			else
333 				MCD_chStatus[channel] = MCD_IDLE;
334 			/* do not change the status if it is already paused. */
335 		}
336 	}
337 	return MCD_chStatus[channel];
338 }
339 
340 /******************** End of MCD_dmaStatus() ************************/
341 
342 /********************************************************************/
343 /* Function:    MCD_startDma
344  * Ppurpose:    Starts a particular kind of DMA
345  * Arguments:
346  * srcAddr	- the channel on which to run the DMA
347  * srcIncr	- the address to move data from, or buffer-descriptor address
348  * destAddr	- the amount to increment the source address per transfer
349  * destIncr	- the address to move data to
350  * dmaSize	- the amount to increment the destination address per transfer
351  * xferSize	- the number bytes in of each data movement (1, 2, or 4)
352  * initiator	- what device initiates the DMA
353  * priority	- priority of the DMA
354  * flags	- flags describing the DMA
355  * funcDesc	- description of byte swapping, bit swapping, and CRC actions
356  * srcAddrVirt	- virtual buffer descriptor address TBD
357  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
358  */
359 
360 int MCD_startDma(int channel, s8 * srcAddr, s16 srcIncr, s8 * destAddr,
361 		 s16 destIncr, u32 dmaSize, u32 xferSize, u32 initiator,
362 		 int priority, u32 flags, u32 funcDesc
363 #ifdef MCD_NEED_ADDR_TRANS
364 		 s8 * srcAddrVirt
365 #endif
366     )
367 {
368 	int srcRsdIncr, destRsdIncr;
369 	int *cSave;
370 	short xferSizeIncr;
371 	int tcrCount = 0;
372 #ifdef MCD_INCLUDE_EU
373 	u32 *realFuncArray;
374 #endif
375 
376 	if ((channel < 0) || (channel >= NCHANNELS))
377 		return (MCD_CHANNEL_INVALID);
378 
379 	/* tbd - need to determine the proper response to a bad funcDesc when
380 	   not including EU functions, for now, assign a benign funcDesc, but
381 	   maybe should return an error */
382 #ifndef MCD_INCLUDE_EU
383 	funcDesc = MCD_FUNC_NOEU1;
384 #endif
385 
386 #ifdef MCD_DEBUG
387 	printf("startDma:Setting up params\n");
388 #endif
389 	/* Set us up for task-wise priority.  We don't technically need to do
390 	   this on every start, but since the register involved is in the same
391 	   longword as other registers that users are in control of, setting
392 	   it more than once is probably preferable.  That since the
393 	   documentation doesn't seem to be completely consistent about the
394 	   nature of the PTD control register. */
395 	MCD_dmaBar->ptdControl |= (u16) 0x8000;
396 
397 	/* Not sure what we need to keep here rtm TBD */
398 #if 1
399 	/* Calculate additional parameters to the regular DMA calls. */
400 	srcRsdIncr = srcIncr < 0 ? -1 : (srcIncr > 0 ? 1 : 0);
401 	destRsdIncr = destIncr < 0 ? -1 : (destIncr > 0 ? 1 : 0);
402 
403 	xferSizeIncr = (xferSize & 0xffff) | 0x20000000;
404 
405 	/* Remember for each channel which variant is running. */
406 	MCD_remVariants.remSrcRsdIncr[channel] = srcRsdIncr;
407 	MCD_remVariants.remDestRsdIncr[channel] = destRsdIncr;
408 	MCD_remVariants.remDestIncr[channel] = destIncr;
409 	MCD_remVariants.remSrcIncr[channel] = srcIncr;
410 	MCD_remVariants.remXferSize[channel] = xferSize;
411 #endif
412 
413 	cSave =
414 	    (int *)(MCD_taskTable[channel].contextSaveSpace) + CSAVE_OFFSET +
415 	    CURRBD;
416 
417 #ifdef MCD_INCLUDE_EU
418 	/* may move this to EU specific calls */
419 	realFuncArray =
420 	    (u32 *) (MCD_taskTable[channel].FDTandFlags & 0xffffff00);
421 	/* Modify the LURC's normal and byte-residue-loop functions according
422 	   to parameter. */
423 	realFuncArray[(LURC * 16)] = xferSize == 4 ?
424 	    funcDesc : xferSize == 2 ?
425 	    funcDesc & 0xfffff00f : funcDesc & 0xffff000f;
426 	realFuncArray[(LURC * 16 + 1)] =
427 	    (funcDesc & MCD_BYTE_SWAP_KILLER) | MCD_NO_BYTE_SWAP_ATALL;
428 #endif
429 	/* Write the initiator field in the TCR, and also set the
430 	   initiator-hold bit. Note that,due to a hardware quirk, this could
431 	   collide with an MDE access to the initiator-register file, so we
432 	   have to verify that the write reads back correctly. */
433 
434 	MCD_dmaBar->taskControl[channel] =
435 	    (initiator << 8) | TASK_CTL_HIPRITSKEN | TASK_CTL_HLDINITNUM;
436 
437 	while (((MCD_dmaBar->taskControl[channel] & 0x1fff) !=
438 		((initiator << 8) | TASK_CTL_HIPRITSKEN | TASK_CTL_HLDINITNUM))
439 	       && (tcrCount < 1000)) {
440 		tcrCount++;
441 		/*MCD_dmaBar->ptd_tcr[channel] = (initiator << 8) | 0x0020; */
442 		MCD_dmaBar->taskControl[channel] =
443 		    (initiator << 8) | TASK_CTL_HIPRITSKEN |
444 		    TASK_CTL_HLDINITNUM;
445 	}
446 
447 	MCD_dmaBar->priority[channel] = (u8) priority & PRIORITY_PRI_MASK;
448 	/* should be albe to handle this stuff with only one write to ts reg
449 	   - tbd */
450 	if (channel < 8 && channel >= 0) {
451 		MCD_dmaBar->taskSize0 &= ~(0xf << (7 - channel) * 4);
452 		MCD_dmaBar->taskSize0 |=
453 		    (xferSize & 3) << (((7 - channel) * 4) + 2);
454 		MCD_dmaBar->taskSize0 |= (xferSize & 3) << ((7 - channel) * 4);
455 	} else {
456 		MCD_dmaBar->taskSize1 &= ~(0xf << (15 - channel) * 4);
457 		MCD_dmaBar->taskSize1 |=
458 		    (xferSize & 3) << (((15 - channel) * 4) + 2);
459 		MCD_dmaBar->taskSize1 |= (xferSize & 3) << ((15 - channel) * 4);
460 	}
461 
462 	/* setup task table flags/options which mostly control the line
463 	   buffers */
464 	MCD_taskTable[channel].FDTandFlags &= ~MCD_TT_FLAGS_MASK;
465 	MCD_taskTable[channel].FDTandFlags |= (MCD_TT_FLAGS_MASK & flags);
466 
467 	if (flags & MCD_FECTX_DMA) {
468 		/* TDTStart and TDTEnd */
469 		MCD_taskTable[channel].TDTstart =
470 		    MCD_modelTaskTable[TASK_FECTX].TDTstart;
471 		MCD_taskTable[channel].TDTend =
472 		    MCD_modelTaskTable[TASK_FECTX].TDTend;
473 		MCD_startDmaENetXmit((char *)srcAddr, (char *)srcAddr,
474 				     (char *)destAddr, MCD_taskTable,
475 				     channel);
476 	} else if (flags & MCD_FECRX_DMA) {
477 		/* TDTStart and TDTEnd */
478 		MCD_taskTable[channel].TDTstart =
479 		    MCD_modelTaskTable[TASK_FECRX].TDTstart;
480 		MCD_taskTable[channel].TDTend =
481 		    MCD_modelTaskTable[TASK_FECRX].TDTend;
482 		MCD_startDmaENetRcv((char *)srcAddr, (char *)srcAddr,
483 				    (char *)destAddr, MCD_taskTable,
484 				    channel);
485 	} else if (flags & MCD_SINGLE_DMA) {
486 		/* this buffer descriptor is used for storing off initial
487 		   parameters for later progress query calculation and for the
488 		   DMA to write the resulting checksum. The DMA does not use
489 		   this to determine how to operate, that info is passed with
490 		   the init routine */
491 		MCD_relocBuffDesc[channel].srcAddr = srcAddr;
492 		MCD_relocBuffDesc[channel].destAddr = destAddr;
493 
494 		/* definitely not its final value */
495 		MCD_relocBuffDesc[channel].lastDestAddr = destAddr;
496 
497 		MCD_relocBuffDesc[channel].dmaSize = dmaSize;
498 		MCD_relocBuffDesc[channel].flags = 0;	/* not used */
499 		MCD_relocBuffDesc[channel].csumResult = 0;	/* not used */
500 		MCD_relocBuffDesc[channel].next = 0;	/* not used */
501 
502 		/* Initialize the progress-querying stuff to show no
503 		   progress: */
504 		((volatile int *)MCD_taskTable[channel].
505 		 contextSaveSpace)[SRCPTR + CSAVE_OFFSET] = (int)srcAddr;
506 		((volatile int *)MCD_taskTable[channel].
507 		 contextSaveSpace)[DESTPTR + CSAVE_OFFSET] = (int)destAddr;
508 		((volatile int *)MCD_taskTable[channel].
509 		 contextSaveSpace)[DCOUNT + CSAVE_OFFSET] = 0;
510 		((volatile int *)MCD_taskTable[channel].
511 		 contextSaveSpace)[CURRBD + CSAVE_OFFSET] =
512 (u32) & (MCD_relocBuffDesc[channel]);
513 		/* tbd - need to keep the user from trying to call the EU
514 		   routine when MCD_INCLUDE_EU is not defined */
515 		if (funcDesc == MCD_FUNC_NOEU1 || funcDesc == MCD_FUNC_NOEU2) {
516 			/* TDTStart and TDTEnd */
517 			MCD_taskTable[channel].TDTstart =
518 			    MCD_modelTaskTable[TASK_SINGLENOEU].TDTstart;
519 			MCD_taskTable[channel].TDTend =
520 			    MCD_modelTaskTable[TASK_SINGLENOEU].TDTend;
521 			MCD_startDmaSingleNoEu((char *)srcAddr, srcIncr,
522 					       (char *)destAddr, destIncr,
523 					       (int)dmaSize, xferSizeIncr,
524 					       flags, (int *)
525 					       &(MCD_relocBuffDesc[channel]),
526 					       cSave, MCD_taskTable, channel);
527 		} else {
528 			/* TDTStart and TDTEnd */
529 			MCD_taskTable[channel].TDTstart =
530 			    MCD_modelTaskTable[TASK_SINGLEEU].TDTstart;
531 			MCD_taskTable[channel].TDTend =
532 			    MCD_modelTaskTable[TASK_SINGLEEU].TDTend;
533 			MCD_startDmaSingleEu((char *)srcAddr, srcIncr,
534 					     (char *)destAddr, destIncr,
535 					     (int)dmaSize, xferSizeIncr,
536 					     flags, (int *)
537 					     &(MCD_relocBuffDesc[channel]),
538 					     cSave, MCD_taskTable, channel);
539 		}
540 	} else {		/* chained DMAS */
541 		/* Initialize the progress-querying stuff to show no
542 		   progress: */
543 #if 1
544 		/* (!defined(MCD_NEED_ADDR_TRANS)) */
545 		((volatile int *)MCD_taskTable[channel].
546 		 contextSaveSpace)[SRCPTR + CSAVE_OFFSET]
547 		    = (int)((MCD_bufDesc *) srcAddr)->srcAddr;
548 		((volatile int *)MCD_taskTable[channel].
549 		 contextSaveSpace)[DESTPTR + CSAVE_OFFSET]
550 		    = (int)((MCD_bufDesc *) srcAddr)->destAddr;
551 #else
552 		/* if using address translation, need the virtual addr of the
553 		   first buffdesc */
554 		((volatile int *)MCD_taskTable[channel].
555 		 contextSaveSpace)[SRCPTR + CSAVE_OFFSET]
556 		    = (int)((MCD_bufDesc *) srcAddrVirt)->srcAddr;
557 		((volatile int *)MCD_taskTable[channel].
558 		 contextSaveSpace)[DESTPTR + CSAVE_OFFSET]
559 		    = (int)((MCD_bufDesc *) srcAddrVirt)->destAddr;
560 #endif
561 		((volatile int *)MCD_taskTable[channel].
562 		 contextSaveSpace)[DCOUNT + CSAVE_OFFSET] = 0;
563 		((volatile int *)MCD_taskTable[channel].
564 		 contextSaveSpace)[CURRBD + CSAVE_OFFSET] = (u32) srcAddr;
565 
566 		if (funcDesc == MCD_FUNC_NOEU1 || funcDesc == MCD_FUNC_NOEU2) {
567 			/*TDTStart and TDTEnd */
568 			MCD_taskTable[channel].TDTstart =
569 			    MCD_modelTaskTable[TASK_CHAINNOEU].TDTstart;
570 			MCD_taskTable[channel].TDTend =
571 			    MCD_modelTaskTable[TASK_CHAINNOEU].TDTend;
572 			MCD_startDmaChainNoEu((int *)srcAddr, srcIncr,
573 					      destIncr, xferSize,
574 					      xferSizeIncr, cSave,
575 					      MCD_taskTable, channel);
576 		} else {
577 			/*TDTStart and TDTEnd */
578 			MCD_taskTable[channel].TDTstart =
579 			    MCD_modelTaskTable[TASK_CHAINEU].TDTstart;
580 			MCD_taskTable[channel].TDTend =
581 			    MCD_modelTaskTable[TASK_CHAINEU].TDTend;
582 			MCD_startDmaChainEu((int *)srcAddr, srcIncr, destIncr,
583 					    xferSize, xferSizeIncr, cSave,
584 					    MCD_taskTable, channel);
585 		}
586 	}
587 	MCD_chStatus[channel] = MCD_IDLE;
588 	return (MCD_OK);
589 }
590 
591 /************************ End of MCD_startDma() *********************/
592 
593 /********************************************************************/
594 /* Function:    MCD_XferProgrQuery
595  * Purpose:     Returns progress of DMA on requested channel
596  * Arguments:   channel - channel to retrieve progress for
597  *              progRep - pointer to user supplied MCD_XferProg struct
598  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
599  *
600  * Notes:
601  *  MCD_XferProgrQuery() upon completing or after aborting a DMA, or
602  *  while the DMA is in progress, this function returns the first
603  *  DMA-destination address not (or not yet) used in the DMA. When
604  *  encountering a non-ready buffer descriptor, the information for
605  *  the last completed descriptor is returned.
606  *
607  *  MCD_XferProgQuery() has to avoid the possibility of getting
608  *  partially-updated information in the event that we should happen
609  *  to query DMA progress just as the DMA is updating it. It does that
610  *  by taking advantage of the fact context is not saved frequently for
611  *  the most part. We therefore read it at least twice until we get the
612  *  same information twice in a row.
613  *
614  *  Because a small, but not insignificant, amount of time is required
615  *  to write out the progress-query information, especially upon
616  *  completion of the DMA, it would be wise to guarantee some time lag
617  *  between successive readings of the progress-query information.
618  */
619 
620 /* How many iterations of the loop below to execute to stabilize values */
621 #define STABTIME 0
622 
623 int MCD_XferProgrQuery(int channel, MCD_XferProg * progRep)
624 {
625 	MCD_XferProg prevRep;
626 	int again;		/* true if we are to try again to ge
627 				   consistent results */
628 	int i;			/* used as a time-waste counter */
629 	int destDiffBytes;	/* Total no of bytes that we think actually
630 				   got xfered. */
631 	int numIterations;	/* number of iterations */
632 	int bytesNotXfered;	/* bytes that did not get xfered. */
633 	s8 *LWAlignedInitDestAddr, *LWAlignedCurrDestAddr;
634 	int subModVal, addModVal;	/* Mode values to added and subtracted
635 					   from the final destAddr */
636 
637 	if ((channel < 0) || (channel >= NCHANNELS))
638 		return (MCD_CHANNEL_INVALID);
639 
640 	/* Read a trial value for the progress-reporting values */
641 	prevRep.lastSrcAddr =
642 	    (s8 *) ((volatile int *)MCD_taskTable[channel].
643 		    contextSaveSpace)[SRCPTR + CSAVE_OFFSET];
644 	prevRep.lastDestAddr =
645 	    (s8 *) ((volatile int *)MCD_taskTable[channel].
646 		    contextSaveSpace)[DESTPTR + CSAVE_OFFSET];
647 	prevRep.dmaSize =
648 	    ((volatile int *)MCD_taskTable[channel].contextSaveSpace)[DCOUNT +
649 								      CSAVE_OFFSET];
650 	prevRep.currBufDesc =
651 	    (MCD_bufDesc *) ((volatile int *)MCD_taskTable[channel].
652 			     contextSaveSpace)[CURRBD + CSAVE_OFFSET];
653 	/* Repeatedly reread those values until they match previous values: */
654 	do {
655 		/* Waste a little bit of time to ensure stability: */
656 		for (i = 0; i < STABTIME; i++) {
657 			/* make sure this loop does something so that it
658 			   doesn't get optimized out */
659 			i += i >> 2;
660 		}
661 		/* Check them again: */
662 		progRep->lastSrcAddr =
663 		    (s8 *) ((volatile int *)MCD_taskTable[channel].
664 			    contextSaveSpace)[SRCPTR + CSAVE_OFFSET];
665 		progRep->lastDestAddr =
666 		    (s8 *) ((volatile int *)MCD_taskTable[channel].
667 			    contextSaveSpace)[DESTPTR + CSAVE_OFFSET];
668 		progRep->dmaSize =
669 		    ((volatile int *)MCD_taskTable[channel].
670 		     contextSaveSpace)[DCOUNT + CSAVE_OFFSET];
671 		progRep->currBufDesc =
672 		    (MCD_bufDesc *) ((volatile int *)MCD_taskTable[channel].
673 				     contextSaveSpace)[CURRBD + CSAVE_OFFSET];
674 		/* See if they match: */
675 		if (prevRep.lastSrcAddr != progRep->lastSrcAddr
676 		    || prevRep.lastDestAddr != progRep->lastDestAddr
677 		    || prevRep.dmaSize != progRep->dmaSize
678 		    || prevRep.currBufDesc != progRep->currBufDesc) {
679 			/* If they don't match, remember previous values and
680 			   try again: */
681 			prevRep.lastSrcAddr = progRep->lastSrcAddr;
682 			prevRep.lastDestAddr = progRep->lastDestAddr;
683 			prevRep.dmaSize = progRep->dmaSize;
684 			prevRep.currBufDesc = progRep->currBufDesc;
685 			again = MCD_TRUE;
686 		} else
687 			again = MCD_FALSE;
688 	} while (again == MCD_TRUE);
689 
690 	/* Update the dCount, srcAddr and destAddr */
691 	/* To calculate dmaCount, we consider destination address. C
692 	   overs M1,P1,Z for destination */
693 	switch (MCD_remVariants.remDestRsdIncr[channel]) {
694 	case MINUS1:
695 		subModVal =
696 		    ((int)progRep->
697 		     lastDestAddr) & ((MCD_remVariants.remXferSize[channel]) -
698 				      1);
699 		addModVal =
700 		    ((int)progRep->currBufDesc->
701 		     destAddr) & ((MCD_remVariants.remXferSize[channel]) - 1);
702 		LWAlignedInitDestAddr =
703 		    (progRep->currBufDesc->destAddr) - addModVal;
704 		LWAlignedCurrDestAddr = (progRep->lastDestAddr) - subModVal;
705 		destDiffBytes = LWAlignedInitDestAddr - LWAlignedCurrDestAddr;
706 		bytesNotXfered =
707 		    (destDiffBytes / MCD_remVariants.remDestIncr[channel]) *
708 		    (MCD_remVariants.remDestIncr[channel]
709 		     + MCD_remVariants.remXferSize[channel]);
710 		progRep->dmaSize =
711 		    destDiffBytes - bytesNotXfered + addModVal - subModVal;
712 		break;
713 	case ZERO:
714 		progRep->lastDestAddr = progRep->currBufDesc->destAddr;
715 		break;
716 	case PLUS1:
717 		/* This value has to be subtracted from the final
718 		   calculated dCount. */
719 		subModVal =
720 		    ((int)progRep->currBufDesc->
721 		     destAddr) & ((MCD_remVariants.remXferSize[channel]) - 1);
722 		/* These bytes are already in lastDestAddr. */
723 		addModVal =
724 		    ((int)progRep->
725 		     lastDestAddr) & ((MCD_remVariants.remXferSize[channel]) -
726 				      1);
727 		LWAlignedInitDestAddr =
728 		    (progRep->currBufDesc->destAddr) - subModVal;
729 		LWAlignedCurrDestAddr = (progRep->lastDestAddr) - addModVal;
730 		destDiffBytes = (progRep->lastDestAddr - LWAlignedInitDestAddr);
731 		numIterations =
732 		    (LWAlignedCurrDestAddr -
733 		     LWAlignedInitDestAddr) /
734 		    MCD_remVariants.remDestIncr[channel];
735 		bytesNotXfered =
736 		    numIterations * (MCD_remVariants.remDestIncr[channel]
737 				     - MCD_remVariants.remXferSize[channel]);
738 		progRep->dmaSize = destDiffBytes - bytesNotXfered - subModVal;
739 		break;
740 	default:
741 		break;
742 	}
743 
744 	/* This covers M1,P1,Z for source */
745 	switch (MCD_remVariants.remSrcRsdIncr[channel]) {
746 	case MINUS1:
747 		progRep->lastSrcAddr =
748 		    progRep->currBufDesc->srcAddr +
749 		    (MCD_remVariants.remSrcIncr[channel] *
750 		     (progRep->dmaSize / MCD_remVariants.remXferSize[channel]));
751 		break;
752 	case ZERO:
753 		progRep->lastSrcAddr = progRep->currBufDesc->srcAddr;
754 		break;
755 	case PLUS1:
756 		progRep->lastSrcAddr =
757 		    progRep->currBufDesc->srcAddr +
758 		    (MCD_remVariants.remSrcIncr[channel] *
759 		     (progRep->dmaSize / MCD_remVariants.remXferSize[channel]));
760 		break;
761 	default:
762 		break;
763 	}
764 
765 	return (MCD_OK);
766 }
767 
768 /******************* End of MCD_XferProgrQuery() ********************/
769 
770 /********************************************************************/
771 /* MCD_resmActions() does the majority of the actions of a DMA resume.
772  * It is called from MCD_killDma() and MCD_resumeDma().  It has to be
773  * a separate function because the kill function has to negate the task
774  * enable before resuming it, but the resume function has to do nothing
775  * if there is no DMA on that channel (i.e., if the enable bit is 0).
776  */
777 static void MCD_resmActions(int channel)
778 {
779 	MCD_dmaBar->debugControl = DBG_CTL_DISABLE;
780 	MCD_dmaBar->debugStatus = MCD_dmaBar->debugStatus;
781 	/* This register is selected to know which initiator is
782 	   actually asserted. */
783 	MCD_dmaBar->ptdDebug = PTD_DBG_TSK_VLD_INIT;
784 
785 	if ((MCD_dmaBar->ptdDebug >> channel) & 0x1)
786 		MCD_chStatus[channel] = MCD_RUNNING;
787 	else
788 		MCD_chStatus[channel] = MCD_IDLE;
789 }
790 
791 /********************* End of MCD_resmActions() *********************/
792 
793 /********************************************************************/
794 /* Function:    MCD_killDma
795  * Purpose:     Halt the DMA on the requested channel, without any
796  *              intention of resuming the DMA.
797  * Arguments:   channel - requested channel
798  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
799  *
800  * Notes:
801  *  A DMA may be killed from any state, including paused state, and it
802  *  always goes to the MCD_HALTED state even if it is killed while in
803  *  the MCD_NO_DMA or MCD_IDLE states.
804  */
805 int MCD_killDma(int channel)
806 {
807 	/* MCD_XferProg progRep; */
808 
809 	if ((channel < 0) || (channel >= NCHANNELS))
810 		return (MCD_CHANNEL_INVALID);
811 
812 	MCD_dmaBar->taskControl[channel] = 0x0;
813 	MCD_resumeDma(channel);
814 	/*
815 	 * This must be after the write to the TCR so that the task doesn't
816 	 * start up again momentarily, and before the status assignment so
817 	 * as to override whatever MCD_resumeDma() may do to the channel
818 	 * status.
819 	 */
820 	MCD_chStatus[channel] = MCD_HALTED;
821 
822 	/*
823 	 * Update the current buffer descriptor's lastDestAddr field
824 	 *
825 	 * MCD_XferProgrQuery (channel, &progRep);
826 	 * progRep.currBufDesc->lastDestAddr = progRep.lastDestAddr;
827 	 */
828 	return (MCD_OK);
829 }
830 
831 /************************ End of MCD_killDma() **********************/
832 
833 /********************************************************************/
834 /* Function:    MCD_continDma
835  * Purpose:     Continue a DMA which as stopped due to encountering an
836  *              unready buffer descriptor.
837  * Arguments:   channel - channel to continue the DMA on
838  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
839  *
840  * Notes:
841  *  This routine does not check to see if there is a task which can
842  *  be continued. Also this routine should not be used with single DMAs.
843  */
844 int MCD_continDma(int channel)
845 {
846 	if ((channel < 0) || (channel >= NCHANNELS))
847 		return (MCD_CHANNEL_INVALID);
848 
849 	MCD_dmaBar->taskControl[channel] |= TASK_CTL_EN;
850 	MCD_chStatus[channel] = MCD_RUNNING;
851 
852 	return (MCD_OK);
853 }
854 
855 /********************** End of MCD_continDma() **********************/
856 
857 /*********************************************************************
858  * MCD_pauseDma() and MCD_resumeDma() below use the DMA's debug unit
859  * to freeze a task and resume it.  We freeze a task by breakpointing
860  * on the stated task.  That is, not any specific place in the task,
861  * but any time that task executes.  In particular, when that task
862  * executes, we want to freeze that task and only that task.
863  *
864  * The bits of the debug control register influence interrupts vs.
865  * breakpoints as follows:
866  * - Bits 14 and 0 enable or disable debug functions.  If enabled, you
867  *   will get the interrupt but you may or may not get a breakpoint.
868  * - Bits 2 and 1 decide whether you also get a breakpoint in addition
869  *   to an interrupt.
870  *
871  * The debug unit can do these actions in response to either internally
872  * detected breakpoint conditions from the comparators, or in response
873  * to the external breakpoint pin, or both.
874  * - Bits 14 and 1 perform the above-described functions for
875  *   internally-generated conditions, i.e., the debug comparators.
876  * - Bits 0 and 2 perform the above-described functions for external
877  *   conditions, i.e., the breakpoint external pin.
878  *
879  * Note that, although you "always" get the interrupt when you turn
880  * the debug functions, the interrupt can nevertheless, if desired, be
881  * masked by the corresponding bit in the PTD's IMR. Note also that
882  * this means that bits 14 and 0 must enable debug functions before
883  * bits 1 and 2, respectively, have any effect.
884  *
885  * NOTE: It's extremely important to not pause more than one DMA channel
886  *  at a time.
887  ********************************************************************/
888 
889 /********************************************************************/
890 /* Function:    MCD_pauseDma
891  * Purpose:     Pauses the DMA on a given channel (if any DMA is running
892  *              on that channel).
893  * Arguments:   channel
894  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
895  */
896 int MCD_pauseDma(int channel)
897 {
898 	/* MCD_XferProg progRep; */
899 
900 	if ((channel < 0) || (channel >= NCHANNELS))
901 		return (MCD_CHANNEL_INVALID);
902 
903 	if (MCD_dmaBar->taskControl[channel] & TASK_CTL_EN) {
904 		MCD_dmaBar->debugComp1 = channel;
905 		MCD_dmaBar->debugControl =
906 		    DBG_CTL_ENABLE | (1 << (channel + 16));
907 		MCD_chStatus[channel] = MCD_PAUSED;
908 
909 		/*
910 		 * Update the current buffer descriptor's lastDestAddr field
911 		 *
912 		 * MCD_XferProgrQuery (channel, &progRep);
913 		 * progRep.currBufDesc->lastDestAddr = progRep.lastDestAddr;
914 		 */
915 	}
916 	return (MCD_OK);
917 }
918 
919 /************************* End of MCD_pauseDma() ********************/
920 
921 /********************************************************************/
922 /* Function:    MCD_resumeDma
923  * Purpose:     Resumes the DMA on a given channel (if any DMA is
924  *              running on that channel).
925  * Arguments:   channel - channel on which to resume DMA
926  * Returns:     MCD_CHANNEL_INVALID if channel is invalid, else MCD_OK
927  */
928 int MCD_resumeDma(int channel)
929 {
930 	if ((channel < 0) || (channel >= NCHANNELS))
931 		return (MCD_CHANNEL_INVALID);
932 
933 	if (MCD_dmaBar->taskControl[channel] & TASK_CTL_EN)
934 		MCD_resmActions(channel);
935 
936 	return (MCD_OK);
937 }
938 
939 /************************ End of MCD_resumeDma() ********************/
940 
941 /********************************************************************/
942 /* Function:    MCD_csumQuery
943  * Purpose:     Provide the checksum after performing a non-chained DMA
944  * Arguments:   channel - channel to report on
945  *              csum - pointer to where to write the checksum/CRC
946  * Returns:     MCD_ERROR if the channel is invalid, else MCD_OK
947  *
948  * Notes:
949  *
950  */
951 int MCD_csumQuery(int channel, u32 * csum)
952 {
953 #ifdef MCD_INCLUDE_EU
954 	if ((channel < 0) || (channel >= NCHANNELS))
955 		return (MCD_CHANNEL_INVALID);
956 
957 	*csum = MCD_relocBuffDesc[channel].csumResult;
958 	return (MCD_OK);
959 #else
960 	return (MCD_ERROR);
961 #endif
962 }
963 
964 /*********************** End of MCD_resumeDma() *********************/
965 
966 /********************************************************************/
967 /* Function:    MCD_getCodeSize
968  * Purpose:     Provide the size requirements of the microcoded tasks
969  * Returns:     Size in bytes
970  */
971 int MCD_getCodeSize(void)
972 {
973 #ifdef MCD_INCLUDE_EU
974 	return (0x2b5c);
975 #else
976 	return (0x173c);
977 #endif
978 }
979 
980 /********************** End of MCD_getCodeSize() ********************/
981 
982 /********************************************************************/
983 /* Function:    MCD_getVersion
984  * Purpose:     Provide the version string and number
985  * Arguments:   longVersion - user supplied pointer to a pointer to a char
986  *                    which points to the version string
987  * Returns:     Version number and version string (by reference)
988  */
989 char MCD_versionString[] = "Multi-channel DMA API Alpha v0.3 (2004-04-26)";
990 #define MCD_REV_MAJOR   0x00
991 #define MCD_REV_MINOR   0x03
992 
993 int MCD_getVersion(char **longVersion)
994 {
995 	*longVersion = MCD_versionString;
996 	return ((MCD_REV_MAJOR << 8) | MCD_REV_MINOR);
997 }
998 
999 /********************** End of MCD_getVersion() *********************/
1000 
1001 /********************************************************************/
1002 /* Private version of memcpy()
1003  * Note that everything this is used for is longword-aligned.
1004  */
1005 static void MCD_memcpy(int *dest, int *src, u32 size)
1006 {
1007 	u32 i;
1008 
1009 	for (i = 0; i < size; i += sizeof(int), dest++, src++)
1010 		*dest = *src;
1011 }
1012