1、杭州电子科技大学 实验报告 课程名称: 嵌入式系统原理课程设计 姓名: 应海东 实验项目: DMA实验、A/D与D/A实验 班级: 10052313 指导教师: 任彧 学号: 10051329 日期: 2012.12.11 实验目的 DMA实验: 了解DMA传送原理; 掌握S3C2410 DMA控制器的使
2、用; 掌握DMA软件编程方法。 A/D与D/A实验: 熟悉S3C2410本身自带的八路十位A/D控制器及相应寄存器; 编程实现ARM 系统的A/D 与D/A功能; 掌握带有A/D 的CPU 编程实现A/D 功能的主要方法; 掌握D/A的实现方法,并使用IIC通信的D/A器件MAX5380。 实验环境 硬件: DM2410实验板; 测控与显示扩展板; PC机; J-Link V8仿真器; 软件: PC机操作系统(WINDOWS XP); ARM Developer Suite v1.2; J-link ARM V4.081; 实验内容(算法、程序、步骤和方法)
3、
DMA实验代码:
1) 头文件、函数声明及特殊功能寄存器
#include
4、void); void DMA_M2M(int ch,int srcAddr,int dstAddr,int tc,int dsz,int burst); //DMA特殊功能寄存器 typedef struct tagDMA { volatile U32 DISRC; //0x0 DMA初始源寄存器 volatile U32 DISRCC; //0x4 DMA初始源控制寄存器 volatile U32 DIDST; //0x8 DMA初始目的寄存器 volatile U32 DIDSTC; //0xc DM
5、A初始目的控制寄存器 volatile U32 DCON; //0x10 DMA控制寄存器 volatile U32 DSTAT; //0x14 DMA状态寄存器 volatile U32 DCSRC; //0x18 当前源寄存器 volatile U32 DCDST; //0x1c 当前目的寄存器 volatile U32 DMASKTRIG; //0x20 DMA掩码触发寄存器 }DMA; static volatile int dmaDone; 2) DMA内存拷贝函数 void DMA_M
6、2M(int ch,int srcAddr,int dstAddr,int tc,int dsz,int burst) { int i,time; volatile U32 memSum0=0,memSum1=0; DMA *pDMA; int length; length = tc*(burst ? 4:1)*((dsz==0)+(dsz==1)*2+(dsz==2)*4); Uart_Printf("[DMA%d MEM2MEM Test]\n",ch); switch(ch) { case 0:
7、 pISR_DMA0 = (int)Dma0Done; rINTMSK &= ~(BIT_DMA0); pDMA = (void *)0x4b000000; //通道0地址 break; case 1: pISR_DMA1 = (int)Dma1Done; rINTMSK &= ~(BIT_DMA1); pDMA = (void *)0x4b000040; //通道1地址 break; case 2: pISR_DMA2 = (int)Dma2Done;
8、rINTMSK &= ~(BIT_DMA2); pDMA = (void *)0x4b000080; //通道2地址 break; case 3: pISR_DMA3 = (int)Dma3Done; rINTMSK &= ~(BIT_DMA3); pDMA = (void *)0x4b0000c0; //通道3地址 break; }
9、 Uart_Printf("DMA%d %8xh->%8xh,size=%xh(tc=%xh),dsz=%d,burst=%d\n",ch, srcAddr,dstAddr,length,tc,dsz,burst); Uart_Printf("Initialize the src.\n"); for(i=srcAddr; i<(srcAddr+length); i+=4) { *((U32 *)i) =
10、i^0x55aa5aa5; memSum0 += i^0x55aa5aa5; } Uart_Printf("DMA%d start\n",ch); dmaDone=0; pDMA->DISRC = srcAddr; pDMA->DISRCC = (0<<1)|(0<<0); // inc,AHB pDMA->DIDST = dstAddr; pDMA->DIDSTC = (0<<1)|(0<<0); // inc,AHB pDMA->DCON = tc|(1<<31)|(1<<30)|(1<<29)|(
11、burst<<28)|(1<<27)|\ (0<<23)|(1<<22)|(dsz<<20)|(tc); //HS,AHB,TC interrupt,whole, SW request mode,relaod off pDMA->DMASKTRIG = (1<<1)|1; //DMA on, SW_TRIG Timer_Start(3);//128us resolution while(dmaDone==0); time = Timer_Stop(); Uart_Printf("DMA transfer
12、 done. time=%f, %fMB/S\n",(float)time/ONESEC3,
length/((float)time/ONESEC3)/1000000.);
rINTMSK = BIT_ALLMSK;
for(i=dstAddr; i 13、emSum0==memSum1)
Uart_Printf("DMA test result--------------------------------------O.K.\n");
else
Uart_Printf("DMA test result--------------------------------------ERROR!!!\n");
}
A/D与D/A实验代码:
#include 14、def.h"
#define REQCNT 100 //May 08, 2002 SOP
#define ADC_FREQ 2500000
//#define ADC_FREQ 1250000
//#define LOOP 1
#define LOOP 10000
void __irq DMA0_Done(void); //Declare Prototype function
void Test_Adc(void);
int ReadAdc(int ch);
void Test_DMA_Adc(void);
int ReadAdc(int ch 15、); //Return type is int, Declare Prototype function
volatile U32 preScaler;
//==========================
//名称:Test_Adc()
//功能:获得A/D输入得电压值
//返回值:void
//参数: void
//=========================
void Test_Adc(void)
{
float a0=0,a1=0,a2=0,a3=0,a4=0,a5=0,a6=0,a7=0; //变量初始化
Uart_Pri 16、ntf("[ ADC_IN Test ]\n");
Uart_Printf("PRESS ANY KEY TO STOP.\n");
preScaler = ADC_FREQ;//
Uart_Printf("ADC conv. freq. = %dHz\n",preScaler);
preScaler = PCLK/ADC_FREQ -1; //PCLK:50.7MHz
//ADC变化时间 5CYCLES*(1/(ADC Freq.)), ADC Freq. = PCLK/(ADCPSR+1)
Uart_P 17、rintf("PCLK/ADC_FREQ - 1 = %d\n",preScaler);
while(!Uart_GetKey())
{
//获得各路输入的值,调用ReadAdc()函数
a0=(float)ReadAdc(0)/1024*3.3*1000;//单位为mV
a1=(float)ReadAdc(1)/1024*3.3*1000;
a2=(float)ReadAdc(2)/1024*3.3*1000;
a3=(float)ReadAdc(3)/1024*3.3*1000;
a 18、4=(float)ReadAdc(4)/1024*3.3*1000;
a5=(float)ReadAdc(5)/1024*3.3*1000;
a6=(float)ReadAdc(6)/1024*3.3*1000;
a7=(float)ReadAdc(7)/1024*3.3*1000;
Uart_Printf("AIN0:%d AIN1:%d AIN2:%d AIN3:%d AIN4:%d AIN5:%d AIN6:%d AIN7:%d\n", (int)a0,(int)a1,(int)a2,(int)a3,(int)a4,(int)a5,(int)a6,( 19、int)a7);
Delay(100);
}
rADCCON=(0<<14)|(19<<6)|(7<<3)|(1<<2); //进入待机模式节约能量消耗
Uart_Printf("TEST FINISHED.\n");
}
//=======================
//名称:ReadAdc()
//功能:读取各路A/D得数值
//返回值:rADCDAT0
//参数: ch
//=========================
int ReadAdc(int ch)
{
int i;
static int prev 20、Ch=-1;
rADCCON = (1<<14)|(preScaler<<6)|(ch<<3); //设置采样通道
if(prevCh!=ch)
{
rADCCON = (1<<14)|(preScaler<<6)|(ch<<3); //设置采样通道
for(i=0;i 21、rADCCON & 0x8000)); //检测是否完成检测
return ( (int)rADCDAT0 & 0x3ff );//返回采到的值
}
实验二:
/******************************************
函数名称:Da_Test
函数功能:驱动DA芯片
返回值:void
参数: void
*****************************************/
void Da_Test (void)
{
unsigned int i,save_E,save_PE;
static U8 data[3] 22、
int ReadAdc(int ch);
float a0=0;
Uart_Printf("[ D/A TEST. ]\n");
Uart_Printf("PRESS ANY KEY TO STOP.\n");
//保护现场
save_E = rGPECON;
save_PE = rGPEUP;
rGPEUP |= 0xc000;
rGPECON |= 0xa0000000;
rGPECON&= 0xafffffff;
rIICCON = (1<<7) | (1<<6) | (1<<5) | (0 23、xf);
rIICADD = 0x10;
rIICSTAT = 0x10;
preScaler = PCLK/ADC_FREQ -1;
//读出DA输出的值
while(!Uart_GetKey())
{for(i=1;i<=255;i++)
{
Da_Out(i);//DA输出电压,1-255对应1-2V
//读出DA输出的值
a0=(float)ReadAdc(0)/1024*3.3*1000;//单位为mV
Uart_Printf("AIN0:%d mV \n",a0);
}
}
Uart_Printf("TEST FINISHED.");
}
结论
当程序运行正常时,在PC机上串口DNW上会显示一连串信息。
实验心得与小结
1. 了解了DMA传送原理并且掌握了DMA软件编程方法。
2. 了解了A/D的切换信息和带有A/D 的CPU 编程实现A/D 功能的主要方法。
3.要动手,没有解决不了的难题,而且可以懂得很多。
指导教师
成绩评定: 指导教师签名:






