主页
FPGA培训
关于平台
企业孵化
峰会大赛
展厅预约
FPGA知识库
联系我们
登录 注册
首页 > 技术文章 > RISC-V嵌入式开发(6):正交编码器
RISC-V嵌入式开发(6):正交编码器
来源: 2021-09-28 10:47

1 概述

上期内容我们介绍了定时器的PWM输出功能,本期内容来介绍一下定时器的正交译码器功能(编码器接口)。正交译码器是和正交编码器外设配合使用的,可对编码器输入的脉冲进行计数进而实现速度测量,本期内容我们通过一个使用旋转编码器的计数小实验,来初步了解它的应用方法。

系统环境:Windows 10 - 64bit

软件平台:PlatformIO IDE 或 NucleiStudio IDE 202102版

硬件需求:RV-STAR开发板、旋转编码器(数字电位器)

2 原理介绍

(1)正交编码器

正交编码器(Quadrature Encoder)是一种用于测量旋转速度和方向的传感器。常见的正交编码器有两个输出信号:A信道和B信道。每个信道可以对运动进行测量并产生数字脉冲,这两个脉冲的相位相差90度(因此称为“正交”),这使得你可以根据它们判断运动的方向,通过积分(累加)运算后,还可以用来测算距离。

上图中,A和B分别连接到两个传感器单元上,黑白相间的圆环称之为「栅格」。传感器单元和栅格的实现方式有很多种,包括「反射式传感器+反光率不同的栅格」「对射式传感器+镂空光栅」「霍尔传感器+磁极圆环」「触点+导轨」等。

本次实验中,我们使用的是下图所示的市面上常见的旋转编码器(数字电位器)。

(2)GD32VF103的正交译码器

正交译码器功能使用TIMERx_CH0和TIMERx_CH1引脚生成的CI0和CI1正交信号各自相互作用产生计数值。通过设置SMC=0x01,0x02或0x03来选择是仅由CI0,仅有CI1,或者由CI0和CI1来决定定时器的计数方向。在每个方向选择源的电平改变期间,DIR位是由硬件自动改变的。计数器计数方向改变的机制如下方的图表所示。

正交译码器可以当作一个带有方向选择的外部时钟,这意味着计数器会在0和自动加载值之间连续地计数。因此,用户必须在计数器开始计数前配置TIMERx_CAR寄存器。

3 实验部分

首先需要参照如下的示意图,对RV-STAR开发板和旋转编码器进行连线:

然后在集成开发环境中创建一个新工程,开始编写代码。

首先需要对定时器的编码器接口进行配置,我们使用的是TIMER2的编码器接口,对应的是PA6和PA7引脚,首先要使能它们的外设时钟和复用时钟,然后配置为浮空输入模式。接着需要创建定时器初始化参数结构体,对定时器的功能进行配置,其中需要注意的是要将结构体参数的预分频系数设为0,周期设为10000(即定时器的自动加载值,也可以设为其他值),然后需要将定时器的模式设置为TIMER_ENCODER_MODE2(编码器模式,使用CI0和CI1计数),然后将定时器的计数值配置为5000(这样读取定时器计数的初值就是5000),最后使能TIMER2。

相关代码实现如下:

void encoder_init(){
    /* TIMER2_CH0 - PA6, TIMER2_CH1 - PA7 */
    rcu_periph_clock_enable(RCU_GPIOA);
    rcu_periph_clock_enable(RCU_AF);
    gpio_init(GPIOA, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, GPIO_PIN_6 | GPIO_PIN_7);

    rcu_periph_clock_enable(RCU_TIMER2);
    timer_deinit(TIMER2);

    /* initialize TIMER init parameter struct */
    timer_parameter_struct timer_initpara;
    timer_struct_para_init(&timer_initpara);
    /* TIMER2 configuration */
    timer_initpara.prescaler         = 0;
    timer_initpara.alignedmode       = TIMER_COUNTER_EDGE;
    timer_initpara.counterdirection  = TIMER_COUNTER_UP;
    timer_initpara.period            = 10000;  /* set auto-reload value */
    timer_initpara.clockdivision     = TIMER_CKDIV_DIV1;
    timer_init(TIMER2, &timer_initpara);

    /* select the encoder mode */
    timer_slave_mode_select(TIMER2, TIMER_ENCODER_MODE2);
    timer_counter_value_config(TIMER2, 5000);  /* config the initial value */

    timer_enable(TIMER2)}

本次实验,我们通过串口的打印输出来查看编码器的计数值,因此在主程序中需要对串口进行初始化,然后在循环体中,每次读取依次定时器的计数值,再打印输出到串口。

int main(){
    encoder_init();
    gd_com_init(GD32_COM0);

    int counter = 0;
    while (1) {
        counter = timer_counter_read(TIMER2);
        printf("Counter: %d\n", counter);

        delay_1ms(500);
    }}

代码编写完成后,进行编译和上传。

然后打开串口终端(波特率115200),可以查看到在串口终端中输出定时器的初值为5000

然后顺时针旋转编码器的旋钮,观察到计数值增加,并且每转动一个单位计数值增加4,符合芯片数据手册中的功能描述

逆时针旋转,计数值减少

以上就是本期实验的全部内容,完整代码请参考链接:timer-encoder-counter · 卡斯帕/rvstar-examples - 码云 - 开源中国 (gitee.com)


声明:文章来源于知乎,本文内容及配图的版权归版权所有人所有,内容仅代表作者个人观点,不代表本网站观点或证实其内容的真实性。对于本网刊载的各类评论非本网评论员评论,仅代表评论者个人观点,并不代表本网证实或赞成其描述。如其他媒体、网站或个人转载使用,需保留本网注明的“稿件来源”,并自负法律责任。本文转载仅为更好的传播行业信息,若有内容图片侵权或者其他问题,请及时通过邮件联系我们,以便做侵删处理。