| 
									
										
										
										
											2009-08-07 20:00:39 +01:00
										 |  |  | /*****************************************************************************
 | 
					
						
							|  |  |  | * Copyright 2004 - 2008 Broadcom Corporation.  All rights reserved. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * Unless you and Broadcom execute a separate written software license | 
					
						
							|  |  |  | * agreement governing use of this software, this software is licensed to you | 
					
						
							|  |  |  | * under the terms of the GNU General Public License version 2, available at | 
					
						
							|  |  |  | * http://www.broadcom.com/licenses/GPLv2.php (the "GPL").
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * Notwithstanding the above, under no circumstances may you combine this | 
					
						
							|  |  |  | * software in any way with any other Broadcom software provided under a | 
					
						
							|  |  |  | * license other than the GPL, without Broadcom's express prior written | 
					
						
							|  |  |  | * consent. | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   @file   dma.h | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @brief  API definitions for the linux DMA interface. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined(ASM_ARM_ARCH_BCMRING_DMA_H)
 | 
					
						
							|  |  |  | #define ASM_ARM_ARCH_BCMRING_DMA_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ---- Include Files ---------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/wait.h>
 | 
					
						
							|  |  |  | #include <linux/semaphore.h>
 | 
					
						
							|  |  |  | #include <csp/dmacHw.h>
 | 
					
						
							|  |  |  | #include <mach/timer.h>
 | 
					
						
							|  |  |  | #include <linux/scatterlist.h>
 | 
					
						
							|  |  |  | #include <linux/dma-mapping.h>
 | 
					
						
							|  |  |  | #include <linux/mm.h>
 | 
					
						
							|  |  |  | #include <linux/vmalloc.h>
 | 
					
						
							|  |  |  | #include <linux/pagemap.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ---- Constants and Types ---------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* If DMA_DEBUG_TRACK_RESERVATION is set to a non-zero value, then the filename */ | 
					
						
							|  |  |  | /* and line number of the reservation request will be recorded in the channel table */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_DEBUG_TRACK_RESERVATION   1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_NUM_CONTROLLERS     2
 | 
					
						
							|  |  |  | #define DMA_NUM_CHANNELS        8	/* per controller */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef enum { | 
					
						
							|  |  |  | 	DMA_DEVICE_MEM_TO_MEM,	/* For memory to memory transfers */ | 
					
						
							|  |  |  | 	DMA_DEVICE_I2S0_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_I2S0_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_I2S1_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_I2S1_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_CODEC_A_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_CODEC_A_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_CODEC_B_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_CODEC_B_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_CODEC_C_DEV_TO_MEM,	/* Additional mic input for beam-forming */ | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_PCM0_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_PCM0_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_PCM1_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_APM_PCM1_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_SPUM_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_SPUM_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_SPIH_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_SPIH_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_UART_A_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_UART_A_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_UART_B_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_UART_B_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_PIF_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_PIF_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_ESW_DEV_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_ESW_MEM_TO_DEV, | 
					
						
							|  |  |  | 	DMA_DEVICE_VPM_MEM_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_CLCD_MEM_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_NAND_MEM_TO_MEM, | 
					
						
							|  |  |  | 	DMA_DEVICE_MEM_TO_VRAM, | 
					
						
							|  |  |  | 	DMA_DEVICE_VRAM_TO_MEM, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Add new entries before this line. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DMA_NUM_DEVICE_ENTRIES, | 
					
						
							|  |  |  | 	DMA_DEVICE_NONE = 0xff,	/* Special value to indicate that no device is currently assigned. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Device_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_Handle_t is the primary object used by callers of the API. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_INVALID_HANDLE  ((DMA_Handle_t) -1)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef int DMA_Handle_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_DescriptorRing_t contains a ring of descriptors which is used | 
					
						
							|  |  |  | *   to point to regions of memory. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	void *virtAddr;		/* Virtual Address of the descriptor ring */ | 
					
						
							|  |  |  | 	dma_addr_t physAddr;	/* Physical address of the descriptor ring */ | 
					
						
							|  |  |  | 	int descriptorsAllocated;	/* Number of descriptors allocated in the descriptor ring */ | 
					
						
							|  |  |  | 	size_t bytesAllocated;	/* Number of bytes allocated in the descriptor ring */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_DescriptorRing_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_MemType_t and DMA_MemMap_t are helper structures used to setup | 
					
						
							|  |  |  | *   DMA chains from a variety of memory sources. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_MEM_MAP_MIN_SIZE    4096	/* Pages less than this size are better */
 | 
					
						
							|  |  |  | 					/* off not being DMA'd. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef enum { | 
					
						
							|  |  |  | 	DMA_MEM_TYPE_NONE,	/* Not a valid setting */ | 
					
						
							|  |  |  | 	DMA_MEM_TYPE_VMALLOC,	/* Memory came from vmalloc call */ | 
					
						
							|  |  |  | 	DMA_MEM_TYPE_KMALLOC,	/* Memory came from kmalloc call */ | 
					
						
							|  |  |  | 	DMA_MEM_TYPE_DMA,	/* Memory came from dma_alloc_xxx call */ | 
					
						
							|  |  |  | 	DMA_MEM_TYPE_USER,	/* Memory came from user space. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_MemType_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* A segment represents a physically and virtually contiguous chunk of memory. */ | 
					
						
							|  |  |  | /* i.e. each segment can be DMA'd */ | 
					
						
							|  |  |  | /* A user of the DMA code will add memory regions. Each region may need to be */ | 
					
						
							|  |  |  | /* represented by one or more segments. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	void *virtAddr;		/* Virtual address used for this segment */ | 
					
						
							|  |  |  | 	dma_addr_t physAddr;	/* Physical address this segment maps to */ | 
					
						
							|  |  |  | 	size_t numBytes;	/* Size of the segment, in bytes */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Segment_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* A region represents a virtually contiguous chunk of memory, which may be */ | 
					
						
							|  |  |  | /* made up of multiple segments. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	DMA_MemType_t memType; | 
					
						
							|  |  |  | 	void *virtAddr; | 
					
						
							|  |  |  | 	size_t numBytes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* Each region (virtually contiguous) consists of one or more segments. Each */ | 
					
						
							|  |  |  | 	/* segment is virtually and physically contiguous. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int numSegmentsUsed; | 
					
						
							|  |  |  | 	int numSegmentsAllocated; | 
					
						
							|  |  |  | 	DMA_Segment_t *segment; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* When a region corresponds to user memory, we need to lock all of the pages */ | 
					
						
							|  |  |  | 	/* down before we can figure out the physical addresses. The lockedPage array contains */ | 
					
						
							|  |  |  | 	/* the pages that were locked, and which subsequently need to be unlocked once the */ | 
					
						
							|  |  |  | 	/* memory is unmapped. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	unsigned numLockedPages; | 
					
						
							|  |  |  | 	struct page **lockedPages; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Region_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	int inUse;		/* Is this mapping currently being used? */ | 
					
						
							|  |  |  | 	struct semaphore lock;	/* Acquired when using this structure */ | 
					
						
							|  |  |  | 	enum dma_data_direction dir;	/* Direction this transfer is intended for */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* In the event that we're mapping user memory, we need to know which task */ | 
					
						
							|  |  |  | 	/* the memory is for, so that we can obtain the correct mm locks. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	struct task_struct *userTask; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	int numRegionsUsed; | 
					
						
							|  |  |  | 	int numRegionsAllocated; | 
					
						
							|  |  |  | 	DMA_Region_t *region; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_MemMap_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_DeviceAttribute_t contains information which describes a | 
					
						
							|  |  |  | *   particular DMA device (or peripheral). | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   It is anticipated that the arrary of DMA_DeviceAttribute_t's will be | 
					
						
							|  |  |  | *   statically initialized. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* The device handler is called whenever a DMA operation completes. The reaon */ | 
					
						
							|  |  |  | /* for it to be called will be a bitmask with one or more of the following bits */ | 
					
						
							|  |  |  | /* set. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_HANDLER_REASON_BLOCK_COMPLETE       dmacHw_INTERRUPT_STATUS_BLOCK
 | 
					
						
							|  |  |  | #define DMA_HANDLER_REASON_TRANSFER_COMPLETE    dmacHw_INTERRUPT_STATUS_TRANS
 | 
					
						
							|  |  |  | #define DMA_HANDLER_REASON_ERROR                dmacHw_INTERRUPT_STATUS_ERROR
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef void (*DMA_DeviceHandler_t) (DMA_Device_t dev, int reason, | 
					
						
							|  |  |  | 				     void *userData); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_ON_DMA0             0x00000001
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_ON_DMA1             0x00000002
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_PORT_PER_DMAC       0x00000004	/* If set, it means that the port used on DMAC0 is different from the port used on DMAC1 */
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_ALLOC_DMA1_FIRST    0x00000008	/* If set, allocate from DMA1 before allocating from DMA0 */
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_IS_DEDICATED        0x00000100
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_NO_ISR              0x00000200
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_ALLOW_LARGE_FIFO    0x00000400
 | 
					
						
							|  |  |  | #define DMA_DEVICE_FLAG_IN_USE              0x00000800	/* If set, device is in use on a channel */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Note: Some DMA devices can be used from multiple DMA Controllers. The bitmask is used to */ | 
					
						
							|  |  |  | /*       determine which DMA controllers a given device can be used from, and the interface */ | 
					
						
							|  |  |  | /*       array determeines the actual interface number to use for a given controller. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	uint32_t flags;		/* Bitmask of DMA_DEVICE_FLAG_xxx constants */ | 
					
						
							|  |  |  | 	uint8_t dedicatedController;	/* Controller number to use if DMA_DEVICE_FLAG_IS_DEDICATED is set. */ | 
					
						
							|  |  |  | 	uint8_t dedicatedChannel;	/* Channel number to use if DMA_DEVICE_FLAG_IS_DEDICATED is set. */ | 
					
						
							|  |  |  | 	const char *name;	/* Will show up in the /proc entry */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uint32_t dmacPort[DMA_NUM_CONTROLLERS];	/* Specifies the port number when DMA_DEVICE_FLAG_PORT_PER_DMAC flag is set */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	dmacHw_CONFIG_t config;	/* Configuration to use when DMA'ing using this device */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void *userData;		/* Passed to the devHandler */ | 
					
						
							|  |  |  | 	DMA_DeviceHandler_t devHandler;	/* Called when DMA operations finish. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	timer_tick_count_t transferStartTime;	/* Time the current transfer was started */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* The following statistical information will be collected and presented in a proc entry. */ | 
					
						
							|  |  |  | 	/* Note: With a contiuous bandwidth of 1 Gb/sec, it would take 584 years to overflow */ | 
					
						
							|  |  |  | 	/*       a 64 bit counter. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uint64_t numTransfers;	/* Number of DMA transfers performed */ | 
					
						
							|  |  |  | 	uint64_t transferTicks;	/* Total time spent doing DMA transfers (measured in timer_tick_count_t's) */ | 
					
						
							|  |  |  | 	uint64_t transferBytes;	/* Total bytes transferred */ | 
					
						
							|  |  |  | 	uint32_t timesBlocked;	/* Number of times a channel was unavailable */ | 
					
						
							|  |  |  | 	uint32_t numBytes;	/* Last transfer size */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* It's not possible to free memory which is allocated for the descriptors from within */ | 
					
						
							|  |  |  | 	/* the ISR. So make the presumption that a given device will tend to use the */ | 
					
						
							|  |  |  | 	/* same sized buffers over and over again, and we keep them around. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DMA_DescriptorRing_t ring;	/* Ring of descriptors allocated for this device */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* We stash away some of the information from the previous transfer. If back-to-back */ | 
					
						
							|  |  |  | 	/* transfers are performed from the same buffer, then we don't have to keep re-initializing */ | 
					
						
							|  |  |  | 	/* the descriptor buffers. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	uint32_t prevNumBytes; | 
					
						
							|  |  |  | 	dma_addr_t prevSrcData; | 
					
						
							|  |  |  | 	dma_addr_t prevDstData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_DeviceAttribute_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   DMA_Channel_t, DMA_Controller_t, and DMA_State_t are really internal | 
					
						
							|  |  |  | *   data structures and don't belong in this header file, but are included | 
					
						
							|  |  |  | *   merely for discussion. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   By the time this is implemented, these structures will be moved out into | 
					
						
							|  |  |  | *   the appropriate C source file instead. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_Channel_t contains state information about each DMA channel. Some | 
					
						
							|  |  |  | *   of the channels are dedicated. Non-dedicated channels are shared | 
					
						
							|  |  |  | *   amongst the other devices. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DMA_CHANNEL_FLAG_IN_USE         0x00000001
 | 
					
						
							|  |  |  | #define DMA_CHANNEL_FLAG_IS_DEDICATED   0x00000002
 | 
					
						
							|  |  |  | #define DMA_CHANNEL_FLAG_NO_ISR         0x00000004
 | 
					
						
							|  |  |  | #define DMA_CHANNEL_FLAG_LARGE_FIFO     0x00000008
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	uint32_t flags;		/* bitmask of DMA_CHANNEL_FLAG_xxx constants */ | 
					
						
							|  |  |  | 	DMA_Device_t devType;	/* Device this channel is currently reserved for */ | 
					
						
							|  |  |  | 	DMA_Device_t lastDevType;	/* Device type that used this previously */ | 
					
						
							|  |  |  | 	char name[20];		/* Name passed onto request_irq */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if (DMA_DEBUG_TRACK_RESERVATION)
 | 
					
						
							|  |  |  | 	const char *fileName;	/* Place where channel reservation took place */ | 
					
						
							|  |  |  | 	int lineNum;		/* Place where channel reservation took place */ | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	dmacHw_HANDLE_t dmacHwHandle;	/* low level channel handle. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Channel_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_Controller_t contains state information about each DMA controller. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The freeChannelQ is stored in the controller data structure rather than | 
					
						
							|  |  |  | *   the channel data structure since several of the devices are accessible | 
					
						
							|  |  |  | *   from multiple controllers, and there is no way to know which controller | 
					
						
							|  |  |  | *   will become available first. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	DMA_Channel_t channel[DMA_NUM_CHANNELS]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Controller_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************
 | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The DMA_Global_t contains all of the global state information used by | 
					
						
							|  |  |  | *   the DMA code. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   Callers which need to allocate a shared channel will be queued up | 
					
						
							|  |  |  | *   on the freeChannelQ until a channel becomes available. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | typedef struct { | 
					
						
							|  |  |  | 	struct semaphore lock;	/* acquired when manipulating table entries */ | 
					
						
							|  |  |  | 	wait_queue_head_t freeChannelQ; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DMA_Controller_t controller[DMA_NUM_CONTROLLERS]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } DMA_Global_t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ---- Variable Externs ------------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | extern DMA_DeviceAttribute_t DMA_gDeviceAttribute[DMA_NUM_DEVICE_ENTRIES]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* ---- Function Prototypes ---------------------------------------------- */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(__KERNEL__)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initializes the DMA module. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       - Success | 
					
						
							|  |  |  | *       < 0     - Error | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_init(void); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if (DMA_DEBUG_TRACK_RESERVATION)
 | 
					
						
							|  |  |  | DMA_Handle_t dma_request_channel_dbg(DMA_Device_t dev, const char *fileName, | 
					
						
							|  |  |  | 				     int lineNum); | 
					
						
							|  |  |  | #define dma_request_channel(dev)  dma_request_channel_dbg(dev, __FILE__, __LINE__)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Reserves a channel for use with @a dev. If the device is setup to use | 
					
						
							|  |  |  | *   a shared channel, then this function will block until a free channel | 
					
						
							|  |  |  | *   becomes available. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       >= 0    - A valid DMA Handle. | 
					
						
							|  |  |  | *       -EBUSY  - Device is currently being used. | 
					
						
							|  |  |  | *       -ENODEV - Device handed in is invalid. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DMA_Handle_t dma_request_channel(DMA_Device_t dev	/* Device to use with the allocated channel. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Frees a previously allocated DMA Handle. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *        0      - DMA Handle was released successfully. | 
					
						
							|  |  |  | *       -EINVAL - Invalid DMA handle | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_free_channel(DMA_Handle_t channel	/* DMA handle. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Determines if a given device has been configured as using a shared | 
					
						
							|  |  |  | *   channel. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return boolean | 
					
						
							|  |  |  | *       0           Device uses a dedicated channel | 
					
						
							|  |  |  | *       non-zero    Device uses a shared channel | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_device_is_channel_shared(DMA_Device_t dev	/* Device to check. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Allocates memory to hold a descriptor ring. The descriptor ring then | 
					
						
							|  |  |  | *   needs to be populated by making one or more calls to | 
					
						
							|  |  |  | *   dna_add_descriptors. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   The returned descriptor ring will be automatically initialized. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0           Descriptor ring was allocated successfully | 
					
						
							|  |  |  | *       -ENOMEM     Unable to allocate memory for the desired number of descriptors. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_alloc_descriptor_ring(DMA_DescriptorRing_t *ring,	/* Descriptor ring to populate */ | 
					
						
							|  |  |  | 			      int numDescriptors	/* Number of descriptors that need to be allocated. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Releases the memory which was previously allocated for a descriptor ring. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dma_free_descriptor_ring(DMA_DescriptorRing_t *ring	/* Descriptor to release */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initializes a descriptor ring, so that descriptors can be added to it. | 
					
						
							|  |  |  | *   Once a descriptor ring has been allocated, it may be reinitialized for | 
					
						
							|  |  |  | *   use with additional/different regions of memory. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   Note that if 7 descriptors are allocated, it's perfectly acceptable to | 
					
						
							|  |  |  | *   initialize the ring with a smaller number of descriptors. The amount | 
					
						
							|  |  |  | *   of memory allocated for the descriptor ring will not be reduced, and | 
					
						
							|  |  |  | *   the descriptor ring may be reinitialized later | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0           Descriptor ring was initialized successfully | 
					
						
							|  |  |  | *       -ENOMEM     The descriptor which was passed in has insufficient space | 
					
						
							|  |  |  | *                   to hold the desired number of descriptors. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_init_descriptor_ring(DMA_DescriptorRing_t *ring,	/* Descriptor ring to initialize */ | 
					
						
							|  |  |  | 			     int numDescriptors	/* Number of descriptors to initialize. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Determines the number of descriptors which would be required for a | 
					
						
							|  |  |  | *   transfer of the indicated memory region. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   This function also needs to know which DMA device this transfer will | 
					
						
							|  |  |  | *   be destined for, so that the appropriate DMA configuration can be retrieved. | 
					
						
							|  |  |  | *   DMA parameters such as transfer width, and whether this is a memory-to-memory | 
					
						
							|  |  |  | *   or memory-to-peripheral, etc can all affect the actual number of descriptors | 
					
						
							|  |  |  | *   required. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       > 0     Returns the number of descriptors required for the indicated transfer | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | *       -ENOMEM Memory exhausted | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_calculate_descriptor_count(DMA_Device_t device,	/* DMA Device that this will be associated with */ | 
					
						
							|  |  |  | 				   dma_addr_t srcData,	/* Place to get data to write to device */ | 
					
						
							|  |  |  | 				   dma_addr_t dstData,	/* Pointer to device data address */ | 
					
						
							|  |  |  | 				   size_t numBytes	/* Number of bytes to transfer to the device */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Adds a region of memory to the descriptor ring. Note that it may take | 
					
						
							|  |  |  | *   multiple descriptors for each region of memory. It is the callers | 
					
						
							|  |  |  | *   responsibility to allocate a sufficiently large descriptor ring. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Descriptors were added successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | *       -ENOMEM Memory exhausted | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_add_descriptors(DMA_DescriptorRing_t *ring,	/* Descriptor ring to add descriptors to */ | 
					
						
							|  |  |  | 			DMA_Device_t device,	/* DMA Device that descriptors are for */ | 
					
						
							|  |  |  | 			dma_addr_t srcData,	/* Place to get data (memory or device) */ | 
					
						
							|  |  |  | 			dma_addr_t dstData,	/* Place to put data (memory or device) */ | 
					
						
							|  |  |  | 			size_t numBytes	/* Number of bytes to transfer to the device */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Sets the descriptor ring associated with a device. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   Once set, the descriptor ring will be associated with the device, even | 
					
						
							|  |  |  | *   across channel request/free calls. Passing in a NULL descriptor ring | 
					
						
							|  |  |  | *   will release any descriptor ring currently associated with the device. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   Note: If you call dma_transfer, or one of the other dma_alloc_ functions | 
					
						
							|  |  |  | *         the descriptor ring may be released and reallocated. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   Note: This function will release the descriptor memory for any current | 
					
						
							|  |  |  | *         descriptor ring associated with this device. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_set_device_descriptor_ring(DMA_Device_t device,	/* Device to update the descriptor ring for. */ | 
					
						
							|  |  |  | 				   DMA_DescriptorRing_t *ring	/* Descriptor ring to add descriptors to */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Retrieves the descriptor ring associated with a device. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_get_device_descriptor_ring(DMA_Device_t device,	/* Device to retrieve the descriptor ring for. */ | 
					
						
							|  |  |  | 				   DMA_DescriptorRing_t *ring	/* Place to store retrieved ring */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Allocates buffers for the descriptors. This is normally done automatically | 
					
						
							|  |  |  | *   but needs to be done explicitly when initiating a dma from interrupt | 
					
						
							|  |  |  | *   context. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Descriptors were allocated successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | *       -ENOMEM Memory exhausted | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_alloc_descriptors(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 			  dmacHw_TRANSFER_TYPE_e transferType,	/* Type of transfer being performed */ | 
					
						
							|  |  |  | 			  dma_addr_t srcData,	/* Place to get data to write to device */ | 
					
						
							|  |  |  | 			  dma_addr_t dstData,	/* Pointer to device data address */ | 
					
						
							|  |  |  | 			  size_t numBytes	/* Number of bytes to transfer to the device */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Allocates and sets up descriptors for a double buffered circular buffer. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   This is primarily intended to be used for things like the ingress samples | 
					
						
							|  |  |  | *   from a microphone. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       > 0     Number of descriptors actually allocated. | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | *       -ENOMEM Memory exhausted | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_alloc_double_dst_descriptors(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 				     dma_addr_t srcData,	/* Physical address of source data */ | 
					
						
							|  |  |  | 				     dma_addr_t dstData1,	/* Physical address of first destination buffer */ | 
					
						
							|  |  |  | 				     dma_addr_t dstData2,	/* Physical address of second destination buffer */ | 
					
						
							|  |  |  | 				     size_t numBytes	/* Number of bytes in each destination buffer */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initializes a DMA_MemMap_t data structure | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_init_mem_map(DMA_MemMap_t *memMap	/* Stores state information about the map */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Releases any memory currently being held by a memory mapping structure. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_term_mem_map(DMA_MemMap_t *memMap	/* Stores state information about the map */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Looks at a memory address and categorizes it. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return One of the values from the DMA_MemType_t enumeration. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DMA_MemType_t dma_mem_type(void *addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Sets the process (aka userTask) associated with a mem map. This is | 
					
						
							|  |  |  | *   required if user-mode segments will be added to the mapping. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void dma_mem_map_set_user_task(DMA_MemMap_t *memMap, | 
					
						
							|  |  |  | 					     struct task_struct *task) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	memMap->userTask = task; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Looks at a memory address and determines if we support DMA'ing to/from | 
					
						
							|  |  |  | *   that type of memory. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return boolean - | 
					
						
							|  |  |  | *               return value != 0 means dma supported | 
					
						
							|  |  |  | *               return value == 0 means dma not supported | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_mem_supports_dma(void *addr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initializes a memory map for use. Since this function acquires a | 
					
						
							|  |  |  | *   sempaphore within the memory map, it is VERY important that dma_unmap | 
					
						
							|  |  |  | *   be called when you're finished using the map. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_map_start(DMA_MemMap_t *memMap,	/* Stores state information about the map */ | 
					
						
							|  |  |  | 		  enum dma_data_direction dir	/* Direction that the mapping will be going */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Adds a segment of memory to a memory map. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return     0 on success, error code otherwise. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_map_add_region(DMA_MemMap_t *memMap,	/* Stores state information about the map */ | 
					
						
							|  |  |  | 		       void *mem,	/* Virtual address that we want to get a map of */ | 
					
						
							|  |  |  | 		       size_t numBytes	/* Number of bytes being mapped */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Creates a descriptor ring from a memory mapping. | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
											
												tree-wide: fix assorted typos all over the place
That is "success", "unknown", "through", "performance", "[re|un]mapping"
, "access", "default", "reasonable", "[con]currently", "temperature"
, "channel", "[un]used", "application", "example","hierarchy", "therefore"
, "[over|under]flow", "contiguous", "threshold", "enough" and others.
Signed-off-by: André Goddard Rosa <andre.goddard@gmail.com>
Signed-off-by: Jiri Kosina <jkosina@suse.cz>
											
										 
											2009-11-14 13:09:05 -02:00
										 |  |  | *   @return 0 on success, error code otherwise. | 
					
						
							| 
									
										
										
										
											2009-08-07 20:00:39 +01:00
										 |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_map_create_descriptor_ring(DMA_Device_t dev,	/* DMA device (where the ring is stored) */ | 
					
						
							|  |  |  | 				   DMA_MemMap_t *memMap,	/* Memory map that will be used */ | 
					
						
							|  |  |  | 				   dma_addr_t devPhysAddr	/* Physical address of device */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Maps in a memory region such that it can be used for performing a DMA. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_map_mem(DMA_MemMap_t *memMap,	/* Stores state information about the map */ | 
					
						
							|  |  |  | 		void *addr,	/* Virtual address that we want to get a map of */ | 
					
						
							|  |  |  | 		size_t count,	/* Number of bytes being mapped */ | 
					
						
							|  |  |  | 		enum dma_data_direction dir	/* Direction that the mapping will be going */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Maps in a memory region such that it can be used for performing a DMA. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_unmap(DMA_MemMap_t *memMap,	/* Stores state information about the map */ | 
					
						
							|  |  |  | 	      int dirtied	/* non-zero if any of the pages were modified */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initiates a transfer when the descriptors have already been setup. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   This is a special case, and normally, the dma_transfer_xxx functions should | 
					
						
							|  |  |  | *   be used. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was started successfully | 
					
						
							|  |  |  | *       -ENODEV Invalid handle | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_start_transfer(DMA_Handle_t handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Stops a previously started DMA transfer. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was stopped successfully | 
					
						
							|  |  |  | *       -ENODEV Invalid handle | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_stop_transfer(DMA_Handle_t handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Waits for a DMA to complete by polling. This function is only intended | 
					
						
							|  |  |  | *   to be used for testing. Interrupts should be used for most DMA operations. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_wait_transfer_done(DMA_Handle_t handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initiates a DMA transfer | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was started successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_transfer(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 		 dmacHw_TRANSFER_TYPE_e transferType,	/* Type of transfer being performed */ | 
					
						
							|  |  |  | 		 dma_addr_t srcData,	/* Place to get data to write to device */ | 
					
						
							|  |  |  | 		 dma_addr_t dstData,	/* Pointer to device data address */ | 
					
						
							|  |  |  | 		 size_t numBytes	/* Number of bytes to transfer to the device */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initiates a transfer from memory to a device. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was started successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _DEV_TO_MEM and not _MEM_TO_DEV) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int dma_transfer_to_device(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 					 dma_addr_t srcData,	/* Place to get data to write to device (physical address) */ | 
					
						
							|  |  |  | 					 dma_addr_t dstData,	/* Pointer to device data address (physical address) */ | 
					
						
							|  |  |  | 					 size_t numBytes	/* Number of bytes to transfer to the device */ | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  | 	return dma_transfer(handle, | 
					
						
							|  |  |  | 			    dmacHw_TRANSFER_TYPE_MEM_TO_PERIPHERAL, | 
					
						
							|  |  |  | 			    srcData, dstData, numBytes); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initiates a transfer from a device to memory. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was started successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device is _MEM_TO_DEV and not _DEV_TO_MEM) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int dma_transfer_from_device(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 					   dma_addr_t srcData,	/* Pointer to the device data address (physical address) */ | 
					
						
							|  |  |  | 					   dma_addr_t dstData,	/* Place to store data retrieved from the device (physical address) */ | 
					
						
							|  |  |  | 					   size_t numBytes	/* Number of bytes to retrieve from the device */ | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  | 	return dma_transfer(handle, | 
					
						
							|  |  |  | 			    dmacHw_TRANSFER_TYPE_PERIPHERAL_TO_MEM, | 
					
						
							|  |  |  | 			    srcData, dstData, numBytes); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Initiates a memory to memory transfer. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       Transfer was started successfully | 
					
						
							|  |  |  | *       -EINVAL Invalid device type for this kind of transfer | 
					
						
							|  |  |  | *               (i.e. the device wasn't DMA_DEVICE_MEM_TO_MEM) | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int dma_transfer_mem_to_mem(DMA_Handle_t handle,	/* DMA Handle */ | 
					
						
							|  |  |  | 					  dma_addr_t srcData,	/* Place to transfer data from (physical address) */ | 
					
						
							|  |  |  | 					  dma_addr_t dstData,	/* Place to transfer data to (physical address) */ | 
					
						
							|  |  |  | 					  size_t numBytes	/* Number of bytes to transfer */ | 
					
						
							|  |  |  |     ) { | 
					
						
							|  |  |  | 	return dma_transfer(handle, | 
					
						
							|  |  |  | 			    dmacHw_TRANSFER_TYPE_MEM_TO_MEM, | 
					
						
							|  |  |  | 			    srcData, dstData, numBytes); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  | *   Set the callback function which will be called when a transfer completes. | 
					
						
							|  |  |  | *   If a NULL callback function is set, then no callback will occur. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @note   @a devHandler will be called from IRQ context. | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | *   @return | 
					
						
							|  |  |  | *       0       - Success | 
					
						
							|  |  |  | *       -ENODEV - Device handed in is invalid. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | /****************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int dma_set_device_handler(DMA_Device_t dev,	/* Device to set the callback for. */ | 
					
						
							|  |  |  | 			   DMA_DeviceHandler_t devHandler,	/* Function to call when the DMA completes */ | 
					
						
							|  |  |  | 			   void *userData	/* Pointer which will be passed to devHandler. */ | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* ASM_ARM_ARCH_BCMRING_DMA_H */
 |