OpenEdv-开源电子网

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

求助!!DMA中断的怪现象!

[复制链接]

2

主题

8

帖子

0

精华

新手上路

积分
23
金钱
23
注册时间
2016-5-9
在线时间
2 小时
发表于 2016-5-9 14:46:19 | 显示全部楼层 |阅读模式
10金钱
使用的STM32F103RBT6,使用DMA将ADC1的规则序列转换结果搬运到RAM中的指定数组。AD的转换是单次扫描转换规则,规则通道有五个。使用定时器每隔一段时间启动转换一次,DMA搬运完成进入中断。配置代码如下:
#include "sys.h"
//启动AD转换器
#define Start_Adc()        ADC1->CR2|=1<<22

//开启一次DMA传输
void DMA_Enable(void)
{
        DMA1_Channel1->CCR&=~(1<<0);                //关闭DMA传输
        DMA1_Channel1->CNDTR=DMA1_MEM_LEN;//DMA1,传输数据量
        DMA1_Channel1->CCR|=1<<0;
}
void Timer2_Init(u16 us)
{
        RCC->APB1ENR|=1<<0;        //TIM2时钟使能
        TIM2->ARR=us-1;                //设定计数器自动重装值//刚好us
        TIM2->PSC=71;                //预分频器71,得到1Mhz的计数时钟
        //这两个东东要同时设置才可以使用中断
        TIM2->DIER|=1<<0;        //允许更新中断
        TIM2->CR1|=0x01;        //使能定时器2
        MY_NVIC_Init(2,1,TIM2_IRQChannel,3);//抢占1,子优先级1,组2
}

void TIM2_IRQHandler(void)
{
        if(TIM2->SR&0x0001)//溢出 中断
        {
                DMA_Enable();
                Start_Adc();
                //确定ADC采样结束
                while(!FG_DMA_TC);
                FG_DMA_TC=0;                        
        }
        TIM2->SR&=~(1<<0);//清中断标志位
}
//初始化ADC
//开启通道0~3、10~11
void Adc_Init(void)
{
                //先初始化IO口
                RCC->APB2ENR|=1<<2;                //使能PORTA口时钟
                GPIOA->CRL&=0XFFFF0000;        //PA0 1 2 3 ANALOG输入
                RCC->APB2ENR|=1<<4;                //使能PORTC口时钟
                GPIOC->CRH&=0XFFFF00FF;        //PC10 11 ANALOG输入

                //ADC1时钟设置
                RCC->APB2ENR|=1<<9;                //ADC1时钟使能
                RCC->APB2RSTR|=1<<9;        //ADC1复位
                RCC->APB2RSTR&=~(1<<9);        //复位结束
                                
                RCC->CFGR&=~(3<<14);        //分频因子清零
                //SYSCLK/DIV2=12M ADC时钟设置为12M,ADC最大时钟不能超过14M!!
                //否则将导致ADC准确度下降
                RCC->CFGR|=2<<14;
               
                //ADC1工作模式设置
                ADC1->CR1&=0XF0FFFF;        //工作模式清零
                ADC1->CR1|=0<<16;                //独立工作模式
                ADC1->CR1|=1<<8;                //扫描模式
                ADC1->CR2&=~(1<<1);                //单次转换模式
                ADC1->CR2&=~(7<<17);
                ADC1->CR2|=7<<17;                //软件控制转换
                ADC1->CR2|=1<<20;                //使用外部触发(SWSTART)!!必须使用一个事件来触发
                ADC1->CR2&=~(1<<11);        //右对齐
                ADC1->CR2|=1<<8;                //使用DMA模式
                                        //设置通道0~3的采样时间
                ADC1->SMPR2=0;
                ADC1->SMPR2|=0xFFFFF6DB;                         //28.5
                                //设置通道10~11的采样时间
                ADC1->SMPR1=0;
                ADC1->SMPR1|=0xFFFFFFDB;                        //28.5
               
                ADC1->SQR1&=~(0xf<<20);        
                ADC1->SQR1|=4<<20;                //5个转换在规则序列中,也就是只转换规则序列1~5
                                  //设置规则转换序列
                ADC1->SQR3&=0XFE000000;        //规则序列清零
                ADC1->SQR3|=ADC_CH0<<0;                  //规则序列1写入
                ADC1->SQR3|=ADC_CH1<<5;                  //规则序列2写入
                ADC1->SQR3|=ADC_CH2<<10;                  //规则序列3写入
                ADC1->SQR3|=ADC_CH3<<15;                  //规则序列4写入
                ADC1->SQR3|=ADC_CH11<<20;                  //规则序列5写入
                //ADC1启动设置
                ADC1->CR2|=1<<0;                //开启AD1转换器
                ADC1->CR2|=1<<3;                //使能复位校准
                while(ADC1->CR2&1<<3);        //等待校准结束
                //该位由软件设置并由硬件清除。在校准寄存器被初始化后该位将被清除。
                ADC1->CR2|=1<<2;                //开启AD1校准
                while(ADC1->CR2&1<<2);        //等待校准结束
                //该位由软件设置以开始校准,并在校准结束时由硬件清除
}

void DMA_Config(void)
{
        u32 paddr;
        paddr = (u32)&ADC1->DR;
        RCC->AHBENR|=1<<0;                                        //开启DMA1时钟
        delay_ms(10);
        DMA1_Channel1->CPAR=paddr;                //DMA1 外设地址
        DMA1_Channel1->CMAR=(u32)ADC_VALUE;                //DMA1 存储器地址
        DMA1_Channel1->CNDTR=DMA1_MEM_LEN;                //DMA1 传输数据量
        DMA1_Channel1->CCR=0x00000000;                        //复位        
        DMA1_Channel1->CCR|=0<<4;                //从外设读
        DMA1_Channel1->CCR|=0<<5;                //普通模式(0:普通模式;1:循环模式)
        DMA1_Channel1->CCR|=0<<6;                //外设地址非增量模式
        DMA1_Channel1->CCR|=1<<7;                //存储器增量模式
        DMA1_Channel1->CCR|=1<<8;                //外设数据宽度为16位
        DMA1_Channel1->CCR|=1<<10;        //存储器数据宽度为16位
        DMA1_Channel1->CCR|=1<<12;        //中等优先级
        DMA1_Channel1->CCR|=0<<14;        //非存储器到存储器模式
        DMA1_Channel1->CCR|=1<<1;                //允许传输完成中断
        MY_NVIC_Init(1,1,DMA1_Channel1_IRQChannel,3);//组2
}
void DMAChannel1_IRQHandler(void)
{
        if(DMA1->ISR&(1<<1))
        {
                FG_DMA_TC = 1;        
                DMA1->IFCR|=1<<1;
        }
}

但是实际运行时有个怪现象,由于我需要AD采用密度比较大,将TIM2的周期定位80us,但是无法进入DMA中断,也就是程序死在TIM2_IRQHandler()中的while(!FG_DMA_TC);
但是将定时器周期改为160us时,程序就能进入DMA中断,程序不再死机。小弟实在想不明白,请各位多多指教,谢谢。

最佳答案

查看完整内容[请看2#楼]

问题已经解决,由于定时器设置的周期太短,在定时器初始化函数中,最后一步是调用MY_NVIC_Init()函数进行优先级设置,但是还来不进行设置完成,定时器中断请求就已经来了,这是定时器中断优先级还是复位默认值0,定时器中断又死等DMA,而这时DMA已经设置好了,优先级为3,比定时器低,所以会被pending,将定时器设置改为如下就可以了。 void Timer2_Init(u16 us) { RCC->APB1ENR|=1PSC=71; //预分频器71,得到1Mhz的计数时钟 ...
正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手上路

积分
23
金钱
23
注册时间
2016-5-9
在线时间
2 小时
 楼主| 发表于 2016-5-9 14:46:20 | 显示全部楼层
问题已经解决,由于定时器设置的周期太短,在定时器初始化函数中,最后一步是调用MY_NVIC_Init()函数进行优先级设置,但是还来不进行设置完成,定时器中断请求就已经来了,这是定时器中断优先级还是复位默认值0,定时器中断又死等DMA,而这时DMA已经设置好了,优先级为3,比定时器低,所以会被pending,将定时器设置改为如下就可以了。
void Timer2_Init(u16 us)
{
        RCC->APB1ENR|=1<<0;        //TIM2时钟使能
        TIM2->ARR=us-1;                //设定计数器自动重装值//刚好us
        TIM2->PSC=71;                //预分频器71,得到1Mhz的计数时钟
        //这两个东东要同时设置才可以使用中断
        MY_NVIC_Init(2,1,TIM2_IRQChannel,3);//抢占1,子优先级1,组2
        TIM2->DIER|=1<<0;        //允许更新中断
        TIM2->CR1|=0x01;        //使能定时器2
}

先进行终端优先级配置,最后使能定时器。
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手上路

积分
23
金钱
23
注册时间
2016-5-9
在线时间
2 小时
 楼主| 发表于 2016-5-9 14:47:53 | 显示全部楼层
本帖最后由 小疯 于 2016-5-9 14:50 编辑

找到另一个现象,当定时器2周期设为80us时,中断优先级并没有设置成功,依然为0,设置成160us时,定时器中断优先级设置成功。导致DMA中断相应不了的原因应该是这个,但是不知道为啥设为80us时优先级设置就不对呢,MY_NVIC_Init()函数是用的原子哥开发板附送的,其他中断设置都没问题。

QQ截图20160509143134.png
QQ截图20160509143451.png
回复

使用道具 举报

6

主题

1097

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3571
金钱
3571
注册时间
2014-12-2
在线时间
365 小时
发表于 2016-5-9 14:49:16 | 显示全部楼层
STM32主频72MHz,你让它每隔80us进一次中断,它实际上已经不堪重负了,中断不能太频繁,中断服务函数必须短小精悍,一般来说STM32的中断频率小于1kHz为宜
回复

使用道具 举报

2

主题

8

帖子

0

精华

新手上路

积分
23
金钱
23
注册时间
2016-5-9
在线时间
2 小时
 楼主| 发表于 2016-5-9 14:54:35 | 显示全部楼层
xkwy 发表于 2016-5-9 14:49
STM32主频72MHz,你让它每隔80us进一次中断,它实际上已经不堪重负了,中断不能太频繁,中断服务函数必须短 ...

是比较频繁,但是确实需要这么办,定时器这么做是可以的,这里的问题是DMA没法进入中断。我不用DMA中断时,只是轮询DMA传输完成标志的话也可以的。问题是我设置80us时,定时器的优先级设置失败,请看2楼的补充。
回复

使用道具 举报

6

主题

1097

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3571
金钱
3571
注册时间
2014-12-2
在线时间
365 小时
发表于 2016-5-9 16:50:25 | 显示全部楼层
小疯 发表于 2016-5-9 14:54
是比较频繁,但是确实需要这么办,定时器这么做是可以的,这里的问题是DMA没法进入中断。我不用DMA中断时 ...

优先级设置不成功就有点儿扯淡了,除非“程序根本没能运行到那”,因为太忙了
坚决不用寄存器,拒绝重复造轮子。
回复

使用道具 举报

6

主题

1097

帖子

0

精华

论坛元老

Rank: 8Rank: 8

积分
3571
金钱
3571
注册时间
2014-12-2
在线时间
365 小时
发表于 2016-5-9 16:51:27 | 显示全部楼层
小疯 发表于 2016-5-9 14:54
是比较频繁,但是确实需要这么办,定时器这么做是可以的,这里的问题是DMA没法进入中断。我不用DMA中断时 ...

要么你改程序架构,要么你换更强的芯片,不过换芯片并不可取,这么频繁的中断任谁也架不住
坚决不用寄存器,拒绝重复造轮子。
回复

使用道具 举报

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

本版积分规则



关闭

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

正点原子公众号

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

GMT+8, 2025-6-9 14:57

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

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