原创 STM32 编码器连接

2009-11-11 15:23 7091 3 4 分类: 电源/新能源
STM32的每个TIMER都有正交编码器输入接口,TI1,TI2经过输入滤波,边沿检测产生TI1FP1,TI2FP2接到编码器模块,通过配置编码器的工作模式,即可以对编码器进行正向/反向计数。

点击看大图



    如下图,编码器使用了A,B两相信号,但是我只需要对TI1信号进行计数(第一行),我也是刚发现了这个错误,原来对两个信号都计数,导致码盘转一周得到不止100个脉冲(100线的光电码盘)。通过STM32的编码器模块比较两想的电平信号就可以很容易地计算出编码器的运行情况了。


点击看大图


下面是我调试OK的代码:


void Encoder_Configration(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;
  TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  TIM_ICInitTypeDef TIM_ICInitStructure;


  //PC6 A相 PC7 B相
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  GPIO_Init(GPIOC,&GPIO_InitStructure);
 
 /* Enable the TIM3 Update Interrupt */
  /*NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQChannel;
  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = TIMx_PRE_EMPTION_PRIORITY;
  NVIC_InitStructure.NVIC_IRQChannelSubPriority = TIMx_SUB_PRIORITY;
  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
  NVIC_Init(&NVIC_InitStructure);*/


  /* Timer configuration in Encoder mode */
 
  TIM_TimeBaseStructure.TIM_Prescaler = 0x0;  // No prescaling
  TIM_TimeBaseStructure.TIM_Period = 10000; 
  TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
  TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;  
  TIM_TimeBaseInit(TIM8, &TIM_TimeBaseStructure);
 
  TIM_EncoderInterfaceConfig(TIM8, TIM_EncoderMode_TI12,
                             TIM_ICPolarity_Rising, TIM_ICPolarity_Rising);
  TIM_ICStructInit(&TIM_ICInitStructure);
  TIM_ICInitStructure.TIM_ICFilter = 6;//ICx_FILTER;
  TIM_ICInit(TIM8, &TIM_ICInitStructure);
 
 // Clear all pending interrupts
  TIM_ClearFlag(TIM8, TIM_FLAG_Update);
  TIM_ITConfig(TIM8, TIM_IT_Update, ENABLE);
  //Reset counter
  TIM2->CNT = 0;
 
  TIM_Cmd(TIM8, ENABLE); 


}


n_Counter = TIM_GetCounter(TIM8);
Diled_Disp_Num((float)n_Counter);


    另外一个值得注意的问题是,STM32 的定时器是16位的,意思是只能计数到65535,有两种方法,一是采用链式的方式用两个定时器将16位扩展为32位,还有一种简单的方法就是开启定时器的溢出中断,每中断一次就代表编码器运转了特定的角度。


    比如编码器是400线的,将ARR寄存器设置为400,每溢出中断一次就代表电机转了一圈,以此类推。


    另外,关于输入脉冲的检测,其实是类似的,只不过在STM32内部是专门用了一个外部触发模块来实现的,如图一中紫色框的标注,编码器模块应该是在这个模块上的升级,下面是配置代码:


void TIM3_ETR_GetDropCounts_Configuration(void)

   GPIO_InitTypeDef GPIO_InitStructure;
   TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
   //test PA0 TIM8_ETR
   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;
   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;;
   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
   GPIO_Init(GPIOD, &GPIO_InitStructure);
  
   TIM_TimeBaseStructure.TIM_Prescaler = 0x00;
   TIM_TimeBaseStructure.TIM_Period = 0xFFFF;
   TIM_TimeBaseStructure.TIM_ClockDivision = 0x0;
   TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
   TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure);  // Time base configuration
 
   TIM_ETRClockMode2Config(TIM3, TIM_ExtTRGPSC_OFF, TIM_ExtTRGPolarity_NonInverted, 0);
 
   TIM_SetCounter(TIM3, 0);
  
   TIM_Cmd(TIM3, ENABLE);
}


转帖》 http://blog.ednchina.com/yemingxp/269451/message.aspx

PARTNER CONTENT

文章评论1条评论)

登录后参与讨论

用户377235 2014-8-14 19:31

时钟也不写,一下tim3 一下tim8,你真测过? RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOE, ENABLE);

相关推荐阅读
用户564291 2009-11-19 10:57
PLC解密
Siemens PPI协议分析    大家好:我是山东临沂的郝金红,PLC解密网是我的个人网站。由于前段时间的疯狂的研究西门子PPI协议解密之故,所以无心插柳的研究出了较实用的西门子S7-200 PP...
用户564291 2009-10-21 16:55
STM32 LCD12864驱动
/****************************************************************【文 件 名 称】lcd12864.h【功 能 描 述】lcd1286...
用户564291 2009-10-17 22:36
DMA控制器
[STM32学习笔记DMA控制器使用 - [ARM]版权声明:转载时请以超链接形式标明文章原始出处和作者信息及本声明http://tigerwang202.blogbus.com/logs/36242...
用户564291 2009-10-16 16:53
成功的调试了一个串口程序
今天调试了一个MDK自带的串口例程,遇到一点小问题:main.c(94): error:  #136: struct "<unnamed>" has no field "USART_Clo...
用户564291 2009-10-16 15:33
我的第一篇日志-ARM学习
                                                       sizeof用法分析1. 定义:sizeof是C/C++中的一个操作符(operator)...
EE直播间
更多
我要评论
1
3
关闭 站长推荐上一条 /3 下一条