OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 5157|回复: 3

adc转换 能够进入dma中断 却不能正确显示数值

[复制链接]

16

主题

47

帖子

0

精华

初级会员

Rank: 2

积分
132
金钱
132
注册时间
2014-4-12
在线时间
0 小时
发表于 2015-5-13 02:39:36 | 显示全部楼层 |阅读模式
5金钱
1原来 是能够正确显示电压的值的 (在没有加dma中断之前)
2加了dma中断后  能够进入dma中断   但显示的值全是0.0000
 3串口的值:
VOL0:0.000000  mv
whyVOL1:0.000000 V
VOL2:0.000000 V
VOL3:0.000000 V
VOL0:0.000000  mv
whyVOL1:0.000000 V
VOL2:0.000000 V
VOL3:0.000000 V
VOL0:0.000000  mv
whyVOL1:0.000000 V
VOL2:0.000000 V
VOL3:0.000000 V

5、dma代码:
[mw_shl_code=c,true]#include "dma.h" #include "delay.h" #include "led.h" #include "usart.h" ////////////////////////////////////////////////////////////////////////////////// //本程序只供学习使用,未经作者许可,不得用于其它任何用途 //ALIENTEK miniSTM32开发板 //DMA 代码 //正点原子@ALIENTEK //技术论坛:www.openedv.com //修改日期:2012/9/8 //版本:V1.0 //版权所有,盗版必究。 //Copyright(C) 广州市星翼电子科技有限公司 2009-2019 //All rights reserved ////////////////////////////////////////////////////////////////////////////////// DMA_InitTypeDef DMA_InitStructure; u16 DMA1_MEM_LEN;//保存DMA每次数据传送的长度 //DMA1的各通道配置 //这里的传输形式是固定的,这点要根据不同的情况来修改 //从存储器->外设模式/8位数据宽度/存储器增量模式 //DMA_CHxMA通道CHx //cpar:外设地址 //cmar:存储器地址 //cndtr:数据传输量 static u16 ADC_ConvertedValue[4]; void MYDMA_Config(DMA_Channel_TypeDef* DMA_CHx,u32 cpar,u32 cmar,u16 cndtr) { NVIC_InitTypeDef NVIC_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE); //使能DMA传输 __nop(); //等待 DMA1 时钟稳定 __nop(); //经测试最少 2 个 nop __nop(); DMA_DeInit(DMA_CHx); //将DMA的通道1寄存器重设为缺省值 DMA1_MEM_LEN=cndtr; DMA_InitStructure.DMA_PeripheralBaseAddr = cpar; //DMA外设ADC基地址 DMA_InitStructure.DMA_MemoryBaseAddr = cmar; //DMA内存基地址 DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //数据传输方向,从waishe读取发送到内存 DMA_InitStructure.DMA_BufferSize = cndtr; //DMA通道的DMA缓存的大小 DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable; //外设地址寄存器不变 DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable; //内存地址寄存器递增 DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //数据宽度为8位 DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord; //数据宽度为8位 DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //工作在循环模式 DMA_InitStructure.DMA_Priority = DMA_Priority_Medium; //DMA通道 x拥有中优先级 DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA通道x没有设置为内存到内存传输 DMA_Init(DMA_CHx, &DMA_InitStructure); //根据DMA_InitStruct中指定的参数初始化DMA的通道USART1_Tx_DMA_Channel所标识的寄存器 NVIC_InitStructure.NVIC_IRQChannel=DMA1_Channel1_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1; NVIC_InitStructure.NVIC_IRQChannelSubPriority=3; NVIC_InitStructure.NVIC_IRQChannelCmd=ENABLE; NVIC_Init(&NVIC_InitStructure); DMA_ITConfig(DMA1_Channel1,DMA_IT_TC,ENABLE);//DMA传输完成中断 DMA_Cmd(DMA1_Channel1,ENABLE); //使能 DMA1 所指示的通道 } void Filter() { } //开启一次DMA传输 void MYDMA_Enable(DMA_Channel_TypeDef*DMA_CHx) { DMA_Cmd(DMA_CHx, DISABLE ); //关闭USART1 TX DMA1 所指示的通道 DMA_SetCurrDataCounter(DMA1_Channel4,DMA1_MEM_LEN);//DMA通道的DMA缓存的大小 获得DMA通道还有多少个数据没有传输 DMA_Cmd(DMA_CHx, ENABLE); //使能USART1 TX DMA1 所指示的通道 } void DMA1_Channel1_IRQHandler() { float Temp0, Temp1,Temp2,Temp3; if(DMA_GetITStatus(DMA1_IT_TC1)) { // Filter();//ADC滤波 DMA_ClearITPendingBit(DMA1_IT_TC1); LED0=!LED0; Temp0=(float)ADC_ConvertedValue[0]*(3.3/4096); Temp1=(float)ADC_ConvertedValue[1]*(3.3/4096); //显示1 Temp2=(float)ADC_ConvertedValue[2]*(3.3/4096); //显示2 Temp3=(float)ADC_ConvertedValue[3]*(3.3/4096); //显示3 delay_ms(500); printf("VOL0:%f mv\r\n",Temp0); // printf("why"); printf("VOL1:%f V\r\n",Temp1); //其余六通道一样处理 //delay_ms(100); printf("VOL2:%f V\r\n",Temp2); //其余六通道一样处理 // delay_ms(100); printf("VOL3:%f V\r\n",Temp3); //其余六四通道一样处理 } } [/mw_shl_code]
6、adc代码
[mw_shl_code=c,true] #include "adc.h" #include "delay.h" ////////////////////////////////////////////////////////////////////////////////// //本程序只供学习使用,未经作者许可,不得用于其它任何用途 //ALIENTEK miniSTM32开发板 //ADC 代码 //正点原子@ALIENTEK //技术论坛:www.openedv.com //修改日期:2012/9/7 //版本:V1.0 //版权所有,盗版必究。 //Copyright(C) 广州市星翼电子科技有限公司 2009-2019 //All rights reserved ////////////////////////////////////////////////////////////////////////////////// //初始化ADC //这里我们仅以规则通道为例 //我们默认将开启通道0~3 void Adc_Init(void) { ADC_InitTypeDef ADC_InitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA |RCC_APB2Periph_ADC1, ENABLE ); //使能ADC1通道时钟 __nop(); __nop(); __nop(); __nop(); RCC_ADCCLKConfig(RCC_PCLK2_Div6); //设置ADC分频因子6 72M/6=12,ADC最大时间不能超过14M //PA1 作为模拟通道输入引脚 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模拟输入引脚 GPIO_Init(GPIOA, &GPIO_InitStructure); ADC_DeInit(ADC1); //复位ADC1,将外设 ADC1 的全部寄存器重设为缺省值 ADC_InitStructure.ADC_Mode = ADC_Mode_Independent; //ADC工作模式:ADC1和ADC2工作在独立模式 ADC_InitStructure.ADC_ScanConvMode = ENABLE; //模数转换工作在扫描模式 ADC_InitStructure.ADC_ContinuousConvMode = ENABLE; //模数转换工作在连续转换模式 ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None; //转换由软件而不是外部触发启动 ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; //ADC数据右对齐 ADC_InitStructure.ADC_NbrOfChannel = 4; //顺序进行规则转换的ADC通道的数目 ADC_Init(ADC1, &ADC_InitStructure); //根据ADC_InitStruct中指定的参数初始化外设ADCx的寄存器 //设置四个通道的采样顺序和采样时间 ADC_RegularChannelConfig(ADC1,ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期 ADC_RegularChannelConfig(ADC1, ADC_Channel_1,2,ADC_SampleTime_239Cycles5 ); ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期 ADC_RegularChannelConfig(ADC1, ADC_Channel_3,4,ADC_SampleTime_239Cycles5 ); ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1 ADC_ResetCalibration(ADC1); //使能复位校准 while(ADC_GetResetCalibrationStatus(ADC1)); //等待复位校准结束 ADC_StartCalibration(ADC1); //开启AD校准 while(ADC_GetCalibrationStatus(ADC1)); //等待校准结束 // ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 } //获得ADC值 //ch:通道值 0~3 u16 Get_Adc() { //设置指定ADC的规则组通道,一个序列,采样时间 ADC_RegularChannelConfig(ADC1,ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期 ADC_RegularChannelConfig(ADC1, ADC_Channel_1,2,ADC_SampleTime_239Cycles5 ); ADC_RegularChannelConfig(ADC1,ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 ); //ADC1,ADC通道,采样时间为239.5周期 ADC_RegularChannelConfig(ADC1, ADC_Channel_3,4,ADC_SampleTime_239Cycles5 ); ADC_SoftwareStartConvCmd(ADC1, ENABLE); //使能指定的ADC1的软件转换启动功能 while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC ));//等待转换结束 return ADC_GetConversionValue(ADC1); //返回最近一次ADC1规则组的转换结果 } u16 Get_Adc_Average(u8 ch,u8 times) { u32 temp_val=0; u8 t; for(t=0;t<times;t++) { temp_val+=Get_Adc( ); delay_ms(5); } return temp_val/times; } [/mw_shl_code]
7main代码
[mw_shl_code=c,true]#include "led.h" #include "delay.h" #include "sys.h" #include "usart.h" #include "dma.h" #include "adc.h" //ADC实验 //注释 真的很重要 //目前为止2015 5 13 dma多通道读取是正常的 //从Temp0=ADC_ConvertedValue[0]*(825>>10); 改成Temp0=(ADC_ConvertedValue[0]*825)>>10;移位计算成功了使用mv 应该会提升计算速度 //进行dma中断试验 2013 5 13 没成功 进入了中断 但是并没有显示数值 不知道为什么 int main(void) { // float Temp1,Temp2,Temp3; // u16 Temp0; // u16 adcx; //float temp; NVIC_Configuration(); delay_init(); //延时函数初始化 uart_init(9600); //串口初始化为9600 LED_Init(); //初始化与LED连接的硬件接口 LED1=0; Adc_Init(); //ADC初始化 MYDMA_Config(DMA1_Channel1,(u32)&ADC1->DR ,(u32)ADC_ConvertedValue,(u32)sizeof(ADC_ConvertedValue)); ADC_SoftwareStartConvCmd(ADC1,ENABLE); //软件启动ADC ADC_DMACmd(ADC1,ENABLE);//开始ADC_DMA传输 LED0=0; while(1) { LED1=!LED1; delay_ms(300); } } [/mw_shl_code]
8 求指点 谢谢!!!

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

13

主题

611

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1929
金钱
1929
注册时间
2014-10-6
在线时间
625 小时
发表于 2015-5-13 07:46:48 | 显示全部楼层
我感觉你的DMA中断中数据还没有来及处理就又进中断了,你一进中断就先把标志清除了,后面还没处理就又进中断了,你在中断用时太长了你分析一下程序会执行到delay_ms(500);后面的语句吗?
回复

使用道具 举报

16

主题

47

帖子

0

精华

初级会员

Rank: 2

积分
132
金钱
132
注册时间
2014-4-12
在线时间
0 小时
 楼主| 发表于 2015-5-14 00:14:52 | 显示全部楼层
回复【2楼】烟酒不沾:
---------------------------------
执行了呀 串口不是打印出来了吗
回复

使用道具 举报

16

主题

47

帖子

0

精华

初级会员

Rank: 2

积分
132
金钱
132
注册时间
2014-4-12
在线时间
0 小时
 楼主| 发表于 2015-5-14 00:28:29 | 显示全部楼层
回复【2楼】烟酒不沾:
---------------------------------
我觉得是内存地址的问题
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-6-22 15:02

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表