历史上的今天
返回首页

历史上的今天

今天是:2024年09月16日(星期一)

2018年09月16日 | stm32编码器的接口模式

2018-09-16 来源:eefocus

编码器是什么玩意呢,它可是一个好玩的东西,做小车测速必不可少的玩意,下面,我将从编码器的原理讲起,一直到用stm32的编码器接口模式,测出电机转速与方向。

1.编码器

图1 编码器示意图

      图1为编码器的示意图,中间是一个带光栅的码盘,光通过光栅,接收管接收到高电平,没通过,接收到低电平。电机旋转一圈,码盘上有多少光栅,接受管就会接收多少个高电平。371电机中的码盘就是这样的,他是334线码盘,具有较高的测速精度,也就是电机转一圈输出334个脉冲,芯片上已集成了脉冲整形触发电路,输出的是矩形波,直接接单片机IO就OK。        
     增量式旋转编码器通过内部两个光敏接受管转化其角度码盘的时序和相位关系,得到其角度码盘角度位移量增加(正方向)或减少(负方向)。下图为编码器的原理图:

图2 增量式旋转编码器

A,B两点对应两个光敏接受管,A,B两点间距为 S2 ,码盘的光栅间距分别为S0和S1。S0+S1的距离是S2的四倍。这样保证了A,B波形相位相差90度。旋转的反向不同,锯齿波A,B先到达高电平的顺序就会不同,如上图左侧所示,顺序的不同,就可以得到旋转的方向。

2.stm32编码器接口模式(寄存器)
        stm32的编码器接口模式在STM32中文参考手册中有详细的说明,在手册273页,14.3.12节。程序是完全按照 下图方式,设置寄存器的。
      

      

图3

         从图3中可以看出,TI1波形先于TI2波形90°时,每遇到一个边沿变化是,计数器加1(可以通过寄存器设置加减),可以看出一个光栅,被计数了4次。TI1波形后于TI2波形90°时 ,每遇到一次边沿变化,计数器减1。


//TIM2_Encoder_Init,Tim2_CH1(PA0);Tim2_CH2(PA1)

//arr:自动重装值 0XFFFF

//psc:时钟预分频数        ,不分频

void TIM2_Encoder_Init(u16 arr,u16 psc)

{

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

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

        

    GPIOA->CRL&=0XFFFFFF00;   //PA0、PA1 清除之前设置

    GPIOA->CRL|=0X00000044;   //PA0、PA1 浮空输入

            

    TIM2->ARR=arr;            //设定计数器自动重装值 

    TIM2->PSC=psc;            //预分频器


        TIM2->CCMR1 |= 1<<0;          //输入模式,IC1FP1映射到TI1上

        TIM2->CCMR1 |= 1<<8;          //输入模式,IC2FP2映射到TI2上

        TIM2->CCER |= 0<<1;                  //IC1不反向

        TIM2->CCER |= 0<<5;           //IC2不反向

        TIM2->SMCR |= 3<<0;                  //所用输入均在上升沿或下降沿有效

        TIM2->CR1 |= 1<<0;                  //使能计数器

}

3 硬件

用到的模块有STM32核心板、L298电机驱动、371带编码器电机(1:34)。这里主要介绍一下电机,1:34指的是电机轴转动34圈,电机输出1圈。1:X,X值越小,电机的输出转速也就越快,扭矩也就越小;反之,X值越大,电机的输出转速越慢,扭矩也越大。


图4 电机实物图

左边两根黄线是电机两极。绿线和白线是脉冲输出线,分别接编码器的接收管A、B,用一根可以测得速度,两根同时用可测出电机速度与转向。红线和黑线是编码器电源接线,红正黑负,电压3.3V-5V,不不可接反。

4 控制代码

工作指示灯、电机方向与速度控制代码。

//LED IO 初始化 端口PD.2  运行指示灯

void LED_Init(void)

{                    

        RCC->APB2ENR|=1<<5;      //使能PORTD时钟         

        GPIOD->CRL&=0XFFFFF0FF;

        GPIOD->CRL|=0X00000300;  //PD.2推挽输出

        GPIOD->ODR|=1<<2;        //PD.2输出高 

}

//电机旋转方向控制信号端口初始化

//PC1~0推挽输出,输出高

void M_Init(void)

{

        RCC->APB2ENR|=1<<4;      //使能PORTC时钟

        GPIOC->CRL&=0XFFFFFF00;

        GPIOC->CRL|=0X00000033;  //PC1~0推挽输出

        GPIOC->ODR|=0XF<<0;             //PC1~0输出高电平

}


//定时器TIM3,PWM输出初始化,CH1(PA6)

//arr:自动重装值

//psc:时钟预分频数

//设置自动重装值为900,那么PWM频率=72000/900=8Khz

////见STM32参考手册,14.3.9PWM模式。

void TIM3_PWM_Init(u16 arr,u16 psc) //arr设定计数器自动重装值   

                                    //psc预分频器不分频,psc=0

{

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

                  

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

        GPIOA->CRL|=0X0B000000;//复用功能输出           

        GPIOA->ODR|=1<<6;//PA6上拉        


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

        TIM3->PSC=psc;//预分频器不分频

        

        TIM3->CCMR1|=6<<4;  //CH1 PWM1模式        高电平有效         

        TIM3->CCMR1|=1<<3; //CH1预装载使能


        TIM3->CCER|=1<<0;   //OC1 输出使能   


        TIM3->CR1=0x0080;   //ARPE使能 

        TIM3->CR1|=0x01;    //使能定时器3 

}


//电机方向与速度控制,速度调节范围为-100~+100

//大于0时,正转,小于0时,反转

// 占空比低于0.4时电机不转

//(占空比是指高电平在一个周期之内所占的时间比率)

//TIM3->CCR1的设定范围为0~900(因为arr=900)

//见STM32参考手册,14.3.9PWM模式。

void Motor_Speed_Control(s16 motorSpeed)         

{

    s16 speed = 0 ;        

        

    if(motorSpeed>100)  speed = 100;

            else if (motorSpeed<-100)  speed = -100;

                        else  speed = motorSpeed;

        if(speed == 0)

        {

                M_1 = 0;

                M_2 = 0;

        }

            else if(speed > 0)

                {

                        M_1 = 0;

                        M_2 = 1;

                        TIM3->CCR1 = speed * 9;

                }

                        else

                        {

                                M_1 = 1;

                                M_2 = 0;

                                TIM3->CCR1 = -speed * 9;

                        }

}


电机速度与方向检测代码


//TIM2_Encoder_Init,Tim2_CH1(PA0);Tim2_CH2(PA1)

//arr:自动重装值 0XFFFF

//psc:时钟预分频数        ,不分频

//见STM32中文手册 14.3.12编码器接口模式

void TIM2_Encoder_Init(u16 arr,u16 psc)

{

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

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

        

    GPIOA->CRL&=0XFFFFFF00;   //PA0、PA1 清除之前设置

    GPIOA->CRL|=0X00000044;   //PA0、PA1 浮空输入

            

    TIM2->ARR=arr;            //设定计数器自动重装值 

    TIM2->PSC=psc;            //预分频器


        TIM2->CCMR1 |= 1<<0;          //输入模式,IC1FP1映射到TI1上

        TIM2->CCMR1 |= 1<<8;          //输入模式,IC2FP2映射到TI2上

        TIM2->CCER |= 0<<1;                  //IC1不反向

        TIM2->CCER |= 0<<5;           //IC2不反向

    TIM2->SMCR |= 3<<0;                  //所用输入均在上升沿或下降沿有效

        TIM2->CR1 |= 1<<0;                  //使能计数器

}

//计数寄存器赋值

void TIM2_Encoder_Write(int data)

{

    TIM2->CNT = data;

}

//读计数个数

int TIM2_Encoder_Read(void)

{

    TIM2_Encoder_Write(0);        //计数器清0

        delay_ms(10);          //检测时间,可调节

    return (int)((s16)(TIM2->CNT));           //数据类型转换

                             //记录边沿变化次数(一个栅格被记录4次)

}

复制代码

这里我们只显示边沿变化次数,没有具体的算出速度。

主函数


int main(void)

{

//        motorSpeed的范围为-100 ~ +100;

        s16 motorSpeed = 100; 

        Stm32_Clock_Init(9); //系统时钟设置

        delay_init(72);             //延时初始化

        uart_init(72,9600);  //串口1初始化

        LED_Init();                         //初始化与LED连接的硬件接口

        M_Init();                           ////初始化电机运行方向控制端口

        TIM3_PWM_Init(900,0);          //不分频。PWM频率=72000/900=8Khz

        TIM2_Encoder_Init(0xffff, 0); //计数器自动重装值为最大


        while(1)

    {

                LED =! LED;

            Motor_Speed_Control(motorSpeed);

                printf("编码器值:%d\n ",TIM2_Encoder_Read());

    }                

}

5 估算验证
         这里我们只是大概的估算验证测量值是否正确,不具有完全正确性。
        我们设置motorSpeed = 100 ,得到测量值如下图:

图5 motorSpeed = 100

因为误差是不可避免的,所以看到每次检测的值都是不一样的。我们取462,因为一个光栅被记录了4次,所以在10ms内一共检测到了462/4=115.5,那么得到11.55个/ms,每ms内检测到11.55个光栅。

     通过码表,记录电机输出50圈,用时50.2s,那么这时应该检测到的光栅个数为50*34(电机转34圈,输出1圈)*334(每圈有334个光栅)=567800,除以时间,得到估算值11.31个/ms。可以看出估算值与测量值是相近的,认为测量是准确的。
      设置motorSpeed = -50 ,得到测量值如下图 :

图6 motorSpeed=-50

可以看到测量值是负值,说明电机是反转,与实际设置相符。

      我们读的是计数器TIM2->CNT中的值,此值为什么会是负的,这里为什么这样用?      编码器模式中使用上下计数,假设我们初始化TIM2_Encoder_Init_1(0xff, 0);自动装载值为0xFF,这时,计数器中的值,就会在0x00与0xFF之间循环变化,由0x01减为0x00,再减1时,计数器中的值为0xFF,我们将此数做为有符型整数处理,当然,计数的前提是每个周期的计数个数不能超过0x7F,超过,计数将不准确。
      符号强制转换,return (int)((s16)(TIM2->CNT));里面有个类型转换,强制转换返回有符型数据。数值都是以补码表示的,正整数补码是源码,负整数补码是绝对值取反加1。向下计数时减1,为0时,就需要向高位借位减“1”,可以这样理解,一个8位数00000000B-00000001B,但0不够减1的,就向不存在的第9位借1,100000000B-00000001B=11111111B,数是以补码形式表示的,这样11111111B就为-1了。
      在例程中,初始化自动重装值为0xFFFF可以做个实验,直接输出TIM2->CNT的值看一下:      printf("编码器值1:%x \r\n",TIM2->CNT)。


推荐阅读

史海拾趣

Colibrys SA公司的发展小趣事

Colibrys在MEMS技术方面拥有深厚的专业知识,这得益于公司长期的研发积累和不断创新的精神。许多客户委托公司建造最苛刻的航空航天、能源、铁路或汽车测试设备,这些成功案例进一步巩固了公司在行业内的领先地位。无论是用于汽车安全气囊的MEMS加速度计,还是用于引导钻头的高温加速度计,都体现了Colibrys在专业知识应用方面的卓越能力。

Hi-G Relays公司的发展小趣事

随着市场的不断扩大和客户需求的多样化,Colibrys逐渐发展成为一站式制造商。公司提供从设计到制造以及最终测试的完整供应链服务,确保客户能够安全、高效地采购到一流性能的MEMS产品。这种服务模式不仅提高了公司的市场竞争力,也赢得了客户的广泛认可和信赖。

Hayashi Denko Co Ltd公司的发展小趣事

Colibrys在MEMS技术方面拥有深厚的专业知识,这得益于公司长期的研发积累和不断创新的精神。许多客户委托公司建造最苛刻的航空航天、能源、铁路或汽车测试设备,这些成功案例进一步巩固了公司在行业内的领先地位。无论是用于汽车安全气囊的MEMS加速度计,还是用于引导钻头的高温加速度计,都体现了Colibrys在专业知识应用方面的卓越能力。

Helium公司的发展小趣事

Helium公司成立于2013年,由Shawn Fanning、Amir Haleem和Sean Carey共同创立。在创立初期,Helium专注于物联网无线网络技术的研究,但并未迅速获得市场关注。这一时期,电子行业正处于创新与转型阶段,物联网作为新兴领域展现出巨大潜力。Helium团队预见到了物联网的发展趋势,但由于缺乏明确的商业模式,其网络发展并未大规模铺开。然而,这段时期的积累为Helium后续的技术突破和市场拓展奠定了坚实基础。

Frequency Management International公司的发展小趣事

FMI深知技术创新是企业发展的核心动力。因此,公司一直致力于在石英晶体和晶体振荡器领域进行技术研发和创新。通过不断的技术积累和优化,FMI成功开发出了一系列高性能、高稳定性的产品,如压控温度补偿晶体振荡器(VCTCXO)、压控晶体振荡器(VCXO)、温度补偿晶体振荡器(TCXO)等。这些产品的推出不仅丰富了FMI的产品线,还进一步提升了公司在行业内的竞争力。

Conexcon Group公司的发展小趣事

在电子行业的发展过程中,环保问题日益受到关注。Conexcon Group积极响应国家环保政策,将绿色环保理念贯穿于产品研发、生产和销售的全过程。公司采用了环保材料和工艺,减少了对环境的污染和破坏。同时,公司还加强了废旧电子产品的回收和处理工作,推动了电子行业的可持续发展。这些举措不仅赢得了消费者的认可和支持,也为公司树立了良好的社会形象。

问答坊 | AI 解惑

让我们DIY一个FPGA开发板,预备中

上周发贴:让我们一起DIY个 FPGA开发板, 报名喽~~~ 非常感谢大家的支持,先分享一些资料,再对电路图做完善后与大家共同探讨! 开发板: 1、先做一个简单的FPGA开发板,小试锋芒。以ALTERA的CYCLONE1为核心,采用EP1C6Q240C8,配合EPC1S。以及LC ...…

查看全部问答>

零起点学习STM32F107开发第2讲( 视频:STM32固件库介绍及使用MDK创建工程)

废话不多说了,请看下边视频。随后的教程我也尽量以视频的形式给出,敬请关注!$(\'swf_Y5X\').innerHTML=AC_FL_RunContent(\'width\', \'550\', \'height\', \'400\', \'allowNetworking\', \'internal\', \'allowScriptAccess\', \'never\', \'sr ...…

查看全部问答>

求对txt文件进行删除行操作代码

本人想把一个几万行的txt文件内的,某些行删除,手工太慢。想把带有某些字符(比如一个单词)的行全部删除。操作以后保存退出。 请问用到什么方法?…

查看全部问答>

vc向evc移植的问题,麻烦高手看一下这段代码!!!

    这是一段Vc代码,意思就是说在一个编辑框(IDC_INPUTTEXT)中输入一句话,点击回车发送出去,同时在本地文本框(IDC_SHOWTEXT)显示出来,我想把这段代码移植到evc,编译时没有问题的,可是下载到开发板上在编辑框(IDC_INPUTTEXT)中输 ...…

查看全部问答>

wince bootloader问题

我购买了一块 yc2440的开发板,硬件构成:s3c2440+64M nandflash+64M ram,其他的都不重要了,反正它是用nandflash启动的。我把开发板内存和nandflash都扩容了一倍,成了128M,现在想自己开发一个bootloader,不知如何下手,请各位大侠指点一下。原 ...…

查看全部问答>

喷墨打印机可以加灌普通碳素墨水吗?

请问:喷墨打印机可以加灌普通碳素墨水吗?谢谢! 我的QQ530481621…

查看全部问答>

问哈关于库连接的问题

   在stdafx.h中加入 //{{包含eSuperMap 类库头文件 #include //}}包含eSuperMap 类库头文件 了后 也在tool---Options加了库文件后 还弹出E:\\eSuperMap\\MyeSuperMap_evc4\\MyeSuperMap_evc4\\stdafx.h(31) : fatal error C1083: Cannot ...…

查看全部问答>

250w功放电源问题

现在在做电机的相关课题,驱动器使用的是普通的功率放大电路,原理上与集成运算放大器相同,目前出现了一个问题: 上周驱动电机时电路正常工作,且带动电机正常运动。经过一段时间后,目前电路不带负载的情况下正常输出波形,但在其带负载后不能正 ...…

查看全部问答>

(++急)有偿请求解决单片机串口多机通信程序问题

各位大侠们,麻烦帮忙看看这个例子,是我毕业设计的题目,由于单片机C编程学得不精,这几天一直在调程序都没法调出来,还请各位大虾们帮帮忙!万分感激!!! 硬件实现功能: 1、从机U2按键按下,在本机显示,同时将按键值送往主机U1,在主机显示 ...…

查看全部问答>

美资企业,招聘 IP protocol engineer(SW 3#)

公司名称: Carrier Access 公司网址: http://www.carrieraccess.com 电子邮箱: lshi@carrieraccess.com,简历请注明信息出处 工作地点: 上海 外语要求: 英文良好 简历接收方式: 英文及中文 学历: 本科以上 职位描述: RESPONSIBILITIES: ...…

查看全部问答>