STM32定时器---正交编码器模式详解

发布者:陈书记最新更新时间:2018-07-19 来源: eefocus关键字:STM32  定时器  正交编码器模式 手机看文章 扫描二维码
随时随地手机看文章

编码器分类: 
按工作原理:光电式、磁电式和触点电刷式 
按码盘的刻孔方式:增量式和绝对式两类 
由于博主接触面还不是很广,一共就用过两个种类的编码器,都是属于光电的 
差分编码器:一般由8根线连接 信号线分别为 A+ A- B+ B- Z+ Z- 以及VCC和GND 
这里有一种不需要Z信号的,6线差分A+ A- B+ B- VCC 和GND 
正交编码器:一般是5根线连接,信号线分别为A B Z VCC和GND

编码器线数: 就是旋转一圈你的A(B)会输出多少个脉冲 ,这里的A B就是上面的输出脉冲信号线,它们转一圈发出的脉冲数一样的,不过存在90°相位差 通常都是360线的 线数越高代表编码器能够反应的位置精度越高

这里写图片描述

相位差为90° 通过判断哪个信号在前 哪个信号在后 来决定TIM->COUNT是++ 还是 – 
360线 AB一圈各为360个,Z信号为一圈一个

编码器信号: 
A 脉冲输出 
B 脉冲输出 
Z 零点信号 当编码器旋转到零点时,Z信号会发出一个脉冲表示现在是零位置 这个零点位置是固定,厂商指定的 
VCC 电源通常分为24V的和5V的 
GND 地线

这里需要注意: 
1.这里的正交编码器是如果是24V的工作电压还需要用光耦隔离,24V转为3V3在接到STM32的定时器两个通道上 
2.脉冲输出是OC门输出,需要上拉电阻 
3.Z信号接到STM32的外部中断口上,很容易受到干扰 ,通常需要接一个电容到GND

这里给出一个24V转3.3V的隔离电路,用到的是6N136光耦

这里写图片描述

硬件连接(这里使用的STM32F103ZET6的TIM4的CH1和CH2): 
PB6–A 
PB7–B 
PA1–Z

这里写图片描述

代码详解: 

TIM4初始化代码如下:


#include "stm32f10x.h"

#include "encode.h"

#include "misc.h"

#include "nvic.h"

#include "sys.h" 

#include "delay.h"


void TIM4_Mode_Config(void)

{

    GPIO_InitTypeDef GPIO_InitStructure;

    TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;

    TIM_ICInitTypeDef TIM_ICInitStructure;      


    //PB6 ch1  A,PB7 ch2 

    RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE);//使能TIM4时钟  

    RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//使能GPIOA时钟


    GPIO_StructInit(&GPIO_InitStructure);//将GPIO_InitStruct中的参数按缺省值输入

    GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;         

    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;//PA6 PA7浮空输入  

    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;

    GPIO_Init(GPIOB, &GPIO_InitStructure);                           


    NVIC_Config(2);


    TIM_DeInit(TIM4);

    TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);

    TIM_TimeBaseStructure.TIM_Period = 359*4;  //设定计数器重装值   TIMx_ARR = 359*4

    TIM_TimeBaseStructure.TIM_Prescaler = 0; //TIM3时钟预分频值

    TIM_TimeBaseStructure.TIM_ClockDivision =TIM_CKD_DIV1 ;//设置时钟分割 T_dts = T_ck_int    

    TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数 

    TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure);              


    TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_BothEdge ,TIM_ICPolarity_BothEdge);//使用编码器模式3,上升下降都计数

    TIM_ICStructInit(&TIM_ICInitStructure);//将结构体中的内容缺省输入

    TIM_ICInitStructure.TIM_ICFilter = 6;  //选择输入比较滤波器 

    TIM_ICInit(TIM4, &TIM_ICInitStructure);//将TIM_ICInitStructure中的指定参数初始化TIM3


//  TIM_ARRPreloadConfig(TIM4, ENABLE);//使能预装载

    TIM_ClearFlag(TIM4, TIM_FLAG_Update);//清除TIM3的更新标志位

    TIM_ITConfig(TIM4, TIM_IT_Update, ENABLE);//运行更新中断

    //Reset counter

    TIM4->CNT = 0;//


    TIM_Cmd(TIM4, ENABLE);   //启动TIM4定时器


}

/*  

void TIM3_Mode_Config(void)

{

    ///TIM3 clock source enable 

    RCC->APB1ENR|=1<<1;       //TIM3时钟使能

    // Enable 1GPIOA, clock 

    RCC->APB2ENR|=1<<2;    //使能PORTA时钟


    // Configure PA.06,07 as encoder input 

    GPIOA->CRL&=0XF0FFFFFF;//PA6

    GPIOA->CRL|=0X04000000;//浮空输入

    GPIOA->CRL&=0X0FFFFFFF;//PA7

    GPIOA->CRL|=0X40000000;//浮空输入


    // Enable the TIM3 Update Interrupt 

    //这两个东东要同时设置才可以使用中断

    TIM3->DIER|=1<<0;   //允许更新中断                

    TIM3->DIER|=1<<6;   //允许触发中断


    TIM3_NVIC_Config();



    //Timer configuration in Encoder mode 

    TIM3->PSC = 0x0;//预分频器

    TIM3->ARR = 15-1;//设定计数器自动重装值 

    TIM3->CR1 &=~(3<<8);// 选择时钟分频:不分频

    TIM3->CR1 &=~(3<<5);// 选择计数模式:边沿对齐模式


    TIM3->CCMR1 |= 1<<0; //CC1S='01' IC1FP1映射到TI1

    TIM3->CCMR1 |= 1<<8; //CC2S='01' IC2FP2映射到TI2

    TIM3->CCER &= ~(1<<1);  //CC1P='0'  IC1FP1不反相,IC1FP1=TI1

    TIM3->CCER &= ~(1<<5);  //CC2P='0'  IC2FP2不反相,IC2FP2=TI2

    TIM3->CCMR1 |= 3<<4; // IC1F='1000' 输入捕获1滤波器

    TIM3->SMCR |= 3<<0;  //SMS='011' 所有的输入均在上升沿和下降沿有效

    TIM3->CNT = 0;

    TIM3->CR1 |= 0x01;    //CEN=1,使能定时器


}*/


void TIM4_Init(void)

{

  TIM4_Mode_Config();

}


这里的NVIC_Config(2)是我个人写的一种多种中断配置的方法单独放在nvic.c中需要了解的可以自己看看工程


这里通常要问的是两点 

1.TIM_TimeBaseStructure.TIM_Period = 359*4 

2.TIM_EncoderInterfaceConfig(TIM4, TIM_EncoderMode_TI12, TIM_ICPolarity_BothEdge ,TIM_ICPolarity_BothEdge);//使用编码器模式3,上升下降都计数


很多人不理解为要360线的编码器为什么这里的重装值乘以4 读出来的为什么又要/4,其实这两个要结合起来解释 

首先看看这个函数TIM_EncoderInterfaceConfig,它有4个参数 

1.选择哪个定时器 即TIM4 

2.编码器模式有三种 见下图 

3.TIM_IC1的极性 

4.TIM_IC2的极性 

这里设置的是编码器模式3,且TI1和TI2都是双边沿触发–即上下边沿都计数

这里写图片描述

还有一个大家不是很懂的图,我来分析一下

这里写图片描述

1.有效边沿 其实就是对应上面设置的编码器的三种模式 
2.相对信号的电平,这里没有理解手册意思,我把它理解为于它的高低电平意味着将PB6和PB7接口对换,PB7接A PB6接B 这样一来就意味着原来的正转变成反转 计数上升变为下降

这里写图片描述

这里写图片描述

TIx 就相当于输入信号的 TIM4->CH1 TIM4->CH2 
TIxF 滤波后信号 
TIxFPx经过带极性选择的边缘检测器过后的产生的信号

3.至于TI1FP1和TI2FP2信号在上身沿计数还是下降沿计数受两点影响 极性(是否反向) 边缘检测(上升沿还是下降沿) 
我们这里设置的是不反向 在双边沿计数,即在A上升下降 B的上身下降都计数

而计数为什么是x4倍 ,下图结合上面的配置详细说明了

这里写图片描述

由此完成了编码器的配置

至于读取编码器角度的时间,要根据实际需要来设置

编码器线数为 w线/圈 
转速为 V 圈/min 
读取间隔时间 t(线间隔时间)

t <= 60/WV 单位为秒

还有Z信号归零,在遇到Z信号的时候,将定时器的CNT=0,这样就能保证位置与CNT实际对应上了 

中断代码如下


//外部中断1,编码器Z相归零  优先级--①  0 0

void EXTI1_IRQHandler(void)

{


    TIM4->CNT = 0;//每次遇到相对零(Z信号)就将计数归0

    TIM_Cmd(TIM4, ENABLE);

    EXTI_ClearITPendingBit(EXTI_Line1);


}



//编码器接口模式     优先级--2   1  1

void TIM4_IRQHandler(void)

{   

    if(TIM4->SR&0x0001)//溢出中断

    {

        ;

    }   

    TIM4->SR&=~(1<<0);//清除中断标志位    

}

1

最后附上工程代码百度云盘跟CSDN下载地址: 

百度云盘:http://pan.baidu.com/s/1bowhzDP 

CSDN:http://download.csdn.net/detail/wang328452854/9417395


关键字:STM32  定时器  正交编码器模式 引用地址:STM32定时器---正交编码器模式详解

上一篇:STM32f429-SDRAM+LTDC总结
下一篇:关于STM32正交编码的问题

推荐阅读最新更新时间:2024-03-16 16:09

STM32成长记之USART--232串口通信
此处介绍最简单的USART使用。 USART基本特性: ● 全双工的,异步通信 ● 分数波特率发生器系统 ─ 发送和接收共用的可编程波特率,最高达4.5Mbits/s ● 可编程数据字长度(8位或9位) ● 可配置的停止位-支持1或2个停止位 ● 检测标志 ─ 接收缓冲器满 ─ 发送缓冲器空 ─ 传输结束标志 ● 校验控制 ─ 发送校验位 ─ 对接收数据进行校验 ● 四个错误检测标志 ─ 溢出错误 ─ 噪音错误 ─ 帧错误 ─ 校验错误 ● 10个带标志的中断源 ─ CTS改变 ─ LIN断开符检测 ─ 发送数据寄存器空 ─ 发送完成 ─ 接收数据寄存器满 ─ 检测到总线为空闲 ─ 溢出错误 ─ 帧错误 ─ 噪
[单片机]
<font color='red'>STM32</font>成长记之USART--232串口通信
STM32学习笔记之正交编码器接口
最近做一个项目,主控芯片用STM32RBT6,要用到光栅尺,本来带一个控制器的,通过控制器的232可以读取光栅尺的数据,但这个控制器太大,设备中放不下,于是,考虑自己做一个,网上看到很多有用CPLD的方案,后来无意间发现stm32的定时器可以配置成编码器,甚喜 高兴之余,突然发现stm32的定时器是16位的,我的光栅尺的计数会超过65535,于是在21ic论坛上和几位高手请教,最终确定的方案 工作过程是配置TIM3为正交编码器模式,并定一个10ms的中断,每10ms读取一次计数值,10ms的前提是在10ms内计数器不溢出(这个思想要感谢21ic的 lxyppc ) 以下是部分代码:(这些代码修改于ST官方的例程,但我的工程用的
[单片机]
STM32串口发送十六进制的数组
通过串口发送十六进制的数组的方法有很多种,这里我介绍一种最简单也最使用的一种方法。 我这里使用的板子是STM32F103RCT6,使用串口2,usart2.c的代码如下: #include delay.h #include usart2.h #include stdarg.h #include stdio.h #include string.h #include timer.h //串口接受缓存区 u8 USART2_RX_BUF ; //接受数据缓冲,最大为USART2_MAX_RECV_LEN个字节 u8 USART2_TX_BUF ; //发送数据缓冲,最大为USART2
[单片机]
窗口看门狗实验——WWDG
概述: 什么是窗口看门狗? 之所以称为窗口就是因为其喂狗时间是一个有上下限的范围内(窗口),你可以通过设定相关寄存器,设定其上限时间(下限固定)。喂狗的时间不能过早也不能过晚。 而独立看门狗限制喂狗时间在0-x内,x由相关寄存器决定。喂狗的时间不能过晚。 为什么要窗口看门狗? 对于一般的看门狗,程序可以在它产生复位前的任意时刻刷新看门狗,但这有一个隐患,有可能程序跑乱了又跑回到正常的地方,或跑乱的程序正好执行了刷新看门狗操作,这样的情况下一般的看门狗就检测不出来了; 如果使用窗口看门狗,程序员可以根据程序正常执行的时间设置刷新看门狗的一个时间窗口,保证不会提前刷新看门狗也不会滞后刷新看门狗,这样可以检测出程序没有按
[单片机]
窗口看门狗实验——WWDG
stm32 待机唤醒实验
void WKUP_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; //EXTI_InitTypeDef EXTI_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_AFIO, ENABLE);//使能GPIOA和复用功能时钟 GPIO_InitStructure.GPIO_Pin =GPIO_Pin_0; //PA.0 GPIO_InitStructure.GPIO_M
[单片机]
STM32单片机通信协议操作步骤及注意事项
STM32单片机通信协议是一种用于在不同设备之间进行数据传输的协议,它可以帮助设备之间进行高效的通信。STM32单片机通信协议可以用于实现多种不同的应用,如智能家居、智能安防、智能交通等。 STM32单片机通信协议的使用方法主要包括以下几个步骤: 1. 首先,需要确定使用的协议类型,如UART、I2C、SPI等,并确定使用的协议的具体参数,如波特率、数据位、停止位等。 2. 然后,需要在STM32单片机上配置相应的通信接口,并将其与外部设备连接起来。 3. 接着,需要编写相应的程序,实现STM32单片机与外部设备之间的数据传输。 4. 最后,需要将程序烧录到STM32单片机上,并运行程序,实现STM32单片机与外部设备之间的数
[单片机]
STM32】HAL库 STM32CubeMX教程五----看门狗
前言: 今天我们来学习看门狗的配置与函数,看门狗可以有效解决程序的跑飞,在使用过程中比较常见,是防止芯片故障的有效外设,我们一起来学习下HAL库 STM32CubeMX的独立看门狗,窗口看门狗的使用。本系列教程将HAL库与STM32CubeMX结合在一起讲解,使您可以更快速的学会各个模块的使用 所用工具: 1、芯片: STM32F407ZET6 2、STM32CubeMx软件 3、IDE: MDK-Keil软件 4、STM32F1xx/STM32F4xxHAL库 知识概括: 通过本篇博客您将学到: STM32CubeMX创建看门狗例程 独立看门狗,靠窗看门狗 工作原理 看门狗 在由单片机
[单片机]
【<font color='red'>STM32</font>】HAL库 STM32CubeMX教程五----看门狗
STM32的一些基本知识总结
Cortex-M3缩略语 AMBA:先进单片机总线架构 ADK:AMBA设计套件 AHB:先进高性能总线 AHB-AP:AHB访问端口 APB:先进外设总线 ARM ARM:ARM架构参考手册 ASIC:行业领域专用集成电路 ATB :先进跟踪总线 BE8:字节不变式大端模式 CPI:每条指令的周期数 DAP:调试访问端口 DSP:数字信号处理(器) DWT:数据观察点及跟踪 ETM:嵌入式跟踪宏单元 FPB:闪存地址重载及断点 FSR:fault状态寄存器 HTM:Core Sight AHB跟踪宏单元 ICE:在线仿真器 IDE:集成开发环境 IRQ:中断请求(通常是外中断请求) ISA:指令系统架构 ISR:中断服务例程 I
[单片机]
<font color='red'>STM32</font>的一些基本知识总结
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved