关于STM32_407输入捕获 脉冲计数 的疑问

2019-07-14 23:44发布

本帖最后由 ligongxiaobie 于 2014-4-22 11:00 编辑

    最近在用STM32做一个脉冲计数的任务,很简单,就是在一定时间内计算出有多少个脉冲,我是通过tiM3进行输入捕获,然后把记到的数放到DMA里,然后用DMA产生中断,在中断里对10组数处理。用的是PB1口。
  但是程序并不是像我想的一样, 在用信号发生器去给一个固定的脉冲的时候,DMA里的10个数并不一样,而且随着脉冲频率的加快,DMA进入中断的速度也加快了,自己感觉代码没有问题,这里附上代码,希望各位前辈批评指正,晚辈感激不尽!
  1. #include "stm32f4xx.h"
  2. #include "hardware_conf.h"
  3. #define TIM3_CCR4_ADDRESS    0x40000440
  4. #define TIM3_CNT_ADDRESS     0x40000424
  5. u16 TIM3_Buffer[10]={0};  //用于缓存计数值
  6.          
  7.          
  8. void TIM3_initial(void)//计数
  9. {
  10.   GPIO_InitTypeDef  GPIO_InitStructure;
  11.   TIM_TimeBaseInitTypeDef TIM_BaseInitStructure;
  12.   TIM_ICInitTypeDef  TIM_ICInitStructure;
  13.   /* TIM1 clock enable */
  14.   RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
  15.   /* GPIOB clock enable */
  16.   RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOB, ENABLE);
  17.   TIM_DeInit(TIM3);//初始化TIM3寄存器

  18.   /* TIM3 channel 3 pin (PE.11) configuration */
  19.   GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_1;
  20.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
  21.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz;
  22.   GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
  23.   GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL;
  24.   GPIO_Init(GPIOB, &GPIO_InitStructure);
  25.   GPIO_PinAFConfig(GPIOB, GPIO_PinSource1, GPIO_AF_TIM3);

  26.   TIM_BaseInitStructure.TIM_Period = 65535;              //周期
  27.   TIM_BaseInitStructure.TIM_Prescaler = 10;          //预分频
  28.   TIM_BaseInitStructure.TIM_ClockDivision = 1;
  29.   TIM_BaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up;
  30.   TIM_TimeBaseInit(TIM3, &TIM_BaseInitStructure);

  31.   TIM_ICInitStructure.TIM_Channel = TIM_Channel_4;
  32.   TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Falling;
  33.   TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI;
  34.   TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;
  35.   TIM_ICInitStructure.TIM_ICFilter = 0x0;
  36.   TIM_ICInit(TIM3, &TIM_ICInitStructure);
  37.   DMA_Cmd(DMA1_Stream2, ENABLE);
  38.   TIM_DMACmd(TIM3, TIM_DMA_CC4, ENABLE);
  39.   /* TIM enable counter */
  40.   TIM_Cmd(TIM3, ENABLE);
  41. }

  42. void TIM3_DMA_Config(void)         //计数的DMA函数
  43. {
  44.   NVIC_InitTypeDef NVIC_InitStructure;
  45.   DMA_InitTypeDef DMA_InitStructure;


  46.   RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_DMA1 , ENABLE);
  47.   DMA_InitStructure.DMA_Channel = DMA_Channel_5;         
  48.   DMA_InitStructure.DMA_PeripheralBaseAddr = (uint32_t)TIM3_CCR4_ADDRESS ;                        
  49.   DMA_InitStructure.DMA_Memory0BaseAddr = (uint32_t)TIM3_Buffer;          //存放计数数组
  50.   DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralToMemory;
  51.   DMA_InitStructure.DMA_BufferSize = 10;
  52.   DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  53.   DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
  54.   DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
  55.   DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
  56.   DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  57.   DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
  58.   DMA_InitStructure.DMA_FIFOMode = DMA_FIFOMode_Enable;
  59.   DMA_InitStructure.DMA_FIFOThreshold = DMA_FIFOThreshold_Full;
  60.   DMA_InitStructure.DMA_MemoryBurst = DMA_MemoryBurst_Single;
  61.   DMA_InitStructure.DMA_PeripheralBurst = DMA_PeripheralBurst_Single;
  62.   DMA_Init(DMA1_Stream2, &DMA_InitStructure);
  63.   DMA_Cmd(DMA1_Stream2, ENABLE);
  64.   DMA_ITConfig(DMA1_Stream2, DMA_IT_TC, ENABLE);//使能DMA中断,在接收完10个数据之后产生中断

  65.   NVIC_InitStructure.NVIC_IRQChannel = DMA1_Stream2_IRQn;
  66.   NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
  67.   NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
  68.   NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  69.   NVIC_Init(&NVIC_InitStructure);
  70.   DMA_Cmd(DMA1_Stream2, ENABLE); //使能DMA1_Steam2
  71. }
复制代码这里是DMA中断里的处理,我这里有个LCD屏,我是在屏上去观察计数的值。
  1. #include "stm32f4xx_it.h"
  2. #include "hardware_conf.h"
  3. #include "DMA_CESU.H"
  4. #include "CeSu_INIT.H"
  5. #include "ili932x_conf.h"

  6. extern u16 Dispaly_Buffer[10];
  7. extern u16 TIM3_Buffer[10];
  8. int c=0;
  9. #define           LED0          PAout(0)
  10. void DMA1_Stream2_IRQHandler(void)         //DMA接受完成中断
  11. {
  12.   u8 i=0,a=0;
  13.   LED0=~LED0;
  14.   /* Test on DMA Stream Transfer Complete interrupt */
  15.   if(DMA_GetITStatus(DMA1_Stream2, DMA_IT_TCIF2))
  16.   {
  17.           for(i=0;i<10;i++)
  18.         {
  19.           Dispaly_Buffer[i]=TIM3_Buffer[i];//将接受到的10个数缓存出来到另一个数组中
  20.         }
  21.         c++;
  22.         if(c==100)          //这里是防止发的数据太快,屏显示不过来
  23.         {               
  24.         c=0;
  25.                 for(a=0;a<10;a++)
  26.                 {
  27.                         Display_4NUM(50,0+a*15,Dispaly_Buffer[a],BLUE);
  28.                         
  29.                 }
  30.         }

  31.   
  32.     DMA_ClearITPendingBit(DMA1_Stream2, DMA_IT_TCIF2);
  33.   }
复制代码


友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
4条回答
z00
1楼-- · 2019-07-15 05:15
最好不使用DMA 计数脉冲完成后停止计数 等DMA处理完成后再进行计数
tjji23
2楼-- · 2019-07-15 09:18
 精彩回答 2  元偷偷看……
349580321@qq.co
3楼-- · 2019-07-15 12:17
谢谢分享谢谢分享谢谢分享谢谢分享谢谢分享谢谢分享
张荣山1990
4楼-- · 2019-07-15 14:47

一周热门 更多>