⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 dma.c

📁 LPCXpresso1768
💻 C
字号:
/****************************************************************************
 *   $Id:: dma.c 5805 2010-12-04 00:33:07Z usb00423                         $
 *   Project: NXP LPC17xx DMA example
 *
 *   Description:
 *     This file contains DMA code example which include DMA 
 *     initialization, DMA interrupt handler, and APIs for DMA
 *     transfer.
 *
 ****************************************************************************
 * Software that is described herein is for illustrative purposes only
 * which provides customers with programming information regarding the
 * products. This software is supplied "AS IS" without any warranties.
 * NXP Semiconductors assumes no responsibility or liability for the
 * use of the software, conveys no license or title under any patent,
 * copyright, or mask work right to the product. NXP Semiconductors
 * reserves the right to make changes in the software without
 * notification. NXP Semiconductors also make no representation or
 * warranty that such application will be suitable for the specified
 * use without further testing or modification.
****************************************************************************/
#include "LPC17xx.h"
#include "type.h"
#include "dma.h"                      
#include "uart_dma.h"

volatile uint32_t DMATCCount = 0;
volatile uint32_t DMAErrCount = 0;
volatile uint32_t UARTDMA0Done = 0;
volatile uint32_t UARTDMA1Done = 0;
volatile uint32_t UARTDMA2Done = 0;
volatile uint32_t UARTDMA3Done = 0;
volatile uint32_t UARTDMA4Done = 0;
volatile uint32_t UARTDMA5Done = 0;
volatile uint32_t UARTDMA6Done = 0;
volatile uint32_t UARTDMA7Done = 0;

/******************************************************************************
** Function name:		DMA_IRQHandler
**
** Descriptions:		DMA interrupt handler
**
** parameters:		None
** Returned value:	None
** 
******************************************************************************/
void DMA_IRQHandler(void) 
{
  uint32_t regVal;

  regVal = LPC_GPDMA->IntTCStat;
  if ( regVal )
  {
	DMATCCount++;
	LPC_GPDMA->IntTCClear = regVal;
	if ( regVal & (0x01<<0) )
	{
	  DMAChannel_Init( 1, P2M );
	  LPC_GPDMACH1->CConfig |= 0x0C001|(DMA_UART1_RX<<1)|(0x00<<6)|(0x02<<11);
	  LPC_UART1->FCR |= 0x08;
	  UARTDMA0Done = 1;
	}
	else if ( regVal & (0x01<<1) )
	{
	  UARTDMA1Done = 1;
	}
	else if ( regVal & (0x01<<2) )
	{
	  DMAChannel_Init( 3, P2M );
	  LPC_GPDMACH3->CConfig |= 0x0C001|(DMA_UART0_RX<<1)|(0x00<<6)|(0x02<<11);
	  LPC_UART0->FCR |= 0x08;
	  UARTDMA2Done = 1;
	}
	else if ( regVal & (0x01<<3) )
	{
	  UARTDMA3Done = 1;
	}
  } 

  regVal = LPC_GPDMA->IntErrStat;
  if ( regVal )
  {
	DMAErrCount++;
	LPC_GPDMA->IntErrClr = regVal;
  }
  return;
}

/******************************************************************************
** Function name:		DMA_Init
**
** Descriptions:		clock to GPDMA in PCONP, GPDMA init before channel init		
**
** parameters:			None			
** Returned value:		None		
** 
******************************************************************************/
void DMA_Init( void )
{
  /* Enable CLOCK into GPDMA controller */
  LPC_SC->PCONP |= (1 << 29);

  /* Select primary function(UART0/1/2/3) in DMA channels,
  secondary is timer 0/1/2/3. */
  LPC_SC->DMAREQSEL = 0x0000;

  LPC_GPDMA->IntTCClear = 0x03;
  LPC_GPDMA->IntErrClr = 0x03;

  LPC_GPDMA->Config = 0x01;	/* Enable DMA channels, little endian */
  while ( !(LPC_GPDMA->Config & 0x01) );

  NVIC_EnableIRQ(DMA_IRQn);
  return;
}

/******************************************************************************
** Function name:		DMAChannel_Init
**
** Descriptions:		
**
** parameters:			Channel number, DMA mode: M2P, P2M, M2M, P2P			
** Returned value:		TRUE or FALSE
** 
******************************************************************************/
uint32_t DMAChannel_Init( uint32_t ChannelNum, uint32_t DMAMode )
{
  if ( ChannelNum == 0 )
  {
	UARTDMA0Done = 0;
	LPC_GPDMA->IntTCClear = 0x01<<0;
	LPC_GPDMA->IntErrClr = 0x01<<0;   
	if ( DMAMode == M2P )
	{
	  /* Ch0 set for M2P transfer from mempry to UART. */
	  LPC_GPDMACH0->CSrcAddr = UART0_DMA_TX_SRC;
	  LPC_GPDMACH0->CDestAddr = UART0_DMA_TX_DST;
	  /* The burst size is set to 1, source and dest transfer width is
	  32 bits(word), Terminal Count Int enable */
	  LPC_GPDMACH0->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 26)|0x80000000;
	}
	else if ( DMAMode == P2M )
	{
	  /* Ch0 set for P2M transfer from UART to memory. */
	  LPC_GPDMACH0->CSrcAddr = UART0_DMA_RX_SRC;
	  LPC_GPDMACH0->CDestAddr = UART0_DMA_RX_DST;
	  /* The burst size is set to 1. Terminal Count Int enable. */
	  LPC_GPDMACH0->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 27)|0x80000000;
	}
	else
	{
	  return ( FALSE );
	}
  }
  else if ( ChannelNum == 1 )
  {
	UARTDMA1Done = 0;   
	LPC_GPDMA->IntTCClear = 0x01<<1;
	LPC_GPDMA->IntErrClr = 0x01<<1;
	if ( DMAMode == M2P )
	{
	  /* Ch1 set for M2P transfer from mempry to UART. */
	  LPC_GPDMACH1->CSrcAddr = UART1_DMA_TX_SRC;
	  LPC_GPDMACH1->CDestAddr = UART1_DMA_TX_DST;
	  /* The burst size is set to 1. Terminal Count Int enable. */
	  LPC_GPDMACH1->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 26)|0x80000000;
	}
	else if ( DMAMode == P2M )
	{
	  /* Ch1 set for P2M transfer from UART to memory. */
	  LPC_GPDMACH1->CSrcAddr = UART1_DMA_RX_SRC;
	  LPC_GPDMACH1->CDestAddr = UART1_DMA_RX_DST;
	  /* The burst size is set to 1. Terminal Count Int enable. */
	  LPC_GPDMACH1->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 27)|0x80000000;
	}
	else
	{
	  return ( FALSE );
	}
  }
  else if ( ChannelNum == 2 )
  {
	UARTDMA2Done = 0;
	LPC_GPDMA->IntTCClear = 0x01<<2;
	LPC_GPDMA->IntErrClr = 0x01<<2;   
	if ( DMAMode == M2P )
	{
	  /* Ch0 set for M2P transfer from mempry to UART. */
	  LPC_GPDMACH2->CSrcAddr = UART1_DMA_TX_SRC;
	  LPC_GPDMACH2->CDestAddr = UART1_DMA_TX_DST;
	  /* The burst size is set to 1, source and dest transfer width is
	  32 bits(word), Terminal Count Int enable */
	  LPC_GPDMACH2->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 26)|0x80000000;
	}
	else if ( DMAMode == P2M )
	{
	  /* Ch0 set for P2M transfer from UART to memory. */
	  LPC_GPDMACH2->CSrcAddr = UART1_DMA_RX_SRC;
	  LPC_GPDMACH2->CDestAddr = UART1_DMA_RX_DST;
	  /* The burst size is set to 1. Terminal Count Int enable. */
	  LPC_GPDMACH2->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 27)|0x80000000;
	}
	else
	{
	  return ( FALSE );
	}
  }
  else if ( ChannelNum == 3 )
  {
	UARTDMA3Done = 0;
	LPC_GPDMA->IntTCClear = 0x01<<3;
	LPC_GPDMA->IntErrClr = 0x01<<3;   
	if ( DMAMode == M2P )
	{
	  /* Ch0 set for M2P transfer from mempry to UART. */
	  LPC_GPDMACH3->CSrcAddr = UART0_DMA_TX_SRC;
	  LPC_GPDMACH3->CDestAddr = UART0_DMA_TX_DST;
	  /* The burst size is set to 1, source and dest transfer width is
	  32 bits(word), Terminal Count Int enable */
	  LPC_GPDMACH3->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 26)|0x80000000;
	}
	else if ( DMAMode == P2M )
	{
	  /* Ch0 set for P2M transfer from UART to memory. */
	  LPC_GPDMACH3->CSrcAddr = UART0_DMA_RX_SRC;
	  LPC_GPDMACH3->CDestAddr = UART0_DMA_RX_DST;
	  /* The burst size is set to 1. Terminal Count Int enable. */
	  LPC_GPDMACH3->CControl = (UART_DMA_SIZE & 0x0FFF)|(0x00 << 12) 
		|(0x00 << 15)|(0x00 << 18)|(0x00 << 21)|(1 << 27)|0x80000000;
	}
	else
	{
	  return ( FALSE );
	}
  }
  else
  {
	return ( FALSE );
  }
  return( TRUE );
}


/******************************************************************************
**                            End Of File
******************************************************************************/

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -