STM32单片机入门教程---TIM编码器接口

前言

编码器测速一般应用在电机控制的项目上,使用PWM驱动电机,再使用编码器测量电机的速度,再用PID算法进行闭环控制,编码器接口模式基本上相当于使用了一个带有方向选择的外部时钟。


编码器接口

简介

  • Encoder Interface 编码器接口
  • 编码器接口可接收增量(正交)编码器的信号,根据编码器旋转产生的正交信号脉冲,自动控制CNT自增或自减,从而指示编码器的位置、旋转方向和旋转速度,编码器接口实际上相当于一个带有方向控制的外部时钟,同时控制CNT的计数时钟和计数方向
  • 每个高级定时器和通用定时器都拥有一个编码器接口;
  • 两个输入引脚借用了输入捕获的通道1和通道2;

正交编码器

正交编码器一般可以测量位置,或者带有方向的速度值,有两个信号输出引脚,A相和B相;
在这里插入图片描述当编码器的旋转轴转起来时,A相和B相会输出方波信号,转的越快方波频率越高,通过测量方波频率即可得知速度,测量方向需要两根线;
当A、B相出现右边两个表的现象时,对应正转或反转;

编码器的设计逻辑:把A相和B相的所有边沿作为计数器的计数时钟,出现边沿信号时,就计数自增或自减,正转的状态都向上计数,反转的状态都向下计数;在这里插入图片描述注意此时在时基单元初始化设置的计数方向并不会使用,因为此时计数方向和计数时钟都处于编码器托管的状态,计数器的自增和自减都受编码器控制;

编码器接口基本结构


输入捕获的前两个通道,通过GPIO口接入编码器的A、B相,然后通过滤波器和边沿检测极性选择,产生TI1FP1和TI2FP2,然后通过编码器接口,编码器接口通过预分频器控制CNT计数器的时钟,同时编码器接口根据编码器的旋转方向,控制CNT的计数方向,一般设置ARR为65535;

工作模式

在这里插入图片描述
仅在TI1或TI2计数表示的意思是,仅在A相或B相的边沿计数,忽略B相或A相的边沿,看一个相的边沿上升下降和另一个相的高低电平的对应关系,即在一个周期看2次对应关系;
而在TI1和TI2计数表示的关系是,A相和B相的边沿计数都要看,即在一个周期看4次对应关系;
一般使用在TI1和TI2上计数的模式,因为精度最高;

实例

均不反相

在这里插入图片描述
编码器抗噪声原理:当方波信号出现毛刺时,按照编码器执行逻辑出现的结果与刚开始时不变,由此达到抗噪声效果;

TI1反相

即边沿检测极性选择中把TI1的方波信号反相处理,在对照信号表时需要把TI1的信号高低电平互换,这样才是反相后实际给编码器接口的电平;
当发现计数方向与我们预期相反时,只要调整一下极性即可;
在这里插入图片描述

代码

编码器接口测速

接线图

在这里插入图片描述

代码思路

  1. RCC开启时钟,开始GPIO和定时器的时钟;
  2. 配置GPIO,把PA6和PA7配置成输入模式;
  3. 配置时基单元,预分频器一般不分频,ARR一般给最大65535;
  4. 配置输入捕获单元;
  5. 配置编码器接口模式;
  6. 调用TIM_Cmd,启动定时器;

库函数

TIM_EncoderInterfaceConfig定时器编码器接口配置;

GPIO模式可以选择上拉、下拉和浮空,通过接在该引脚的外部模块输出的默认电平来确定选择哪种模式,和外部模块保持默认状态一致,防止默认电平打架;
如果外部模块空闲默认输出高电平,选择上拉输入,默认输入高电平;如果外部模块空闲默认输出低电平,选择下拉输入,默认输入低电平;

编码器接口会托管时钟,所以这里不需要选择时钟源,编码器接口就是一个带方向的外部时钟;

输入捕获单元只使用了通道1和通道2的滤波器和极性选择,这里的极性选择中上升沿代表的是高低电平不反转;
在这里插入图片描述

总结

本节内容对应手册的14.3.12编码器接口模式。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要使用STM32C8T6和JGB37-520编码器电机进行测距,您需要进行以下步骤: 1. 连接JGB37-520编码器电机和STM32C8T6单片机。 2. 配置STM32C8T6的定时器,以便读取编码器信号。 3. 通过读取编码器信号,计算电机的转速和位置。 4. 使用电机的转速和位置信息,计算电机的线速度和加速度。 5. 将电机的线速度和加速度转换为距离。 下面是一些可能有帮助的代码片段: 1. 连接JGB37-520编码器电机和STM32C8T6单片机 JGB37-520编码器电机有两个信号线,一个是A相信号线,一个是B相信号线。将A相信号线连接到STM32C8T6的TIMx_CH1引脚,将B相信号线连接到TIMx_CH2引脚。 2. 配置STM32C8T6的定时器,以便读取编码器信号 使用STM32的定时器来读取编码器信号。您需要配置定时器的输入捕获模式,以便捕获编码器信号的上升沿和下降沿。您还需要设置定时器的计数器和预分频器,以便在每个捕获事件之间测量时间。下面是一个示例配置: ```c // 定义定时器和GPIO引脚 #define TIMx TIM2 #define TIMx_CLK RCC_APB1Periph_TIM2 #define TIMx_IRQn TIM2_IRQn #define TIMx_IRQHandler TIM2_IRQHandler #define TIMx_CH1_GPIO_PORT GPIOA #define TIMx_CH1_GPIO_PIN GPIO_Pin_0 #define TIMx_CH1_GPIO_CLK RCC_AHB1Periph_GPIOA #define TIMx_CH1_SOURCE GPIO_PinSource0 #define TIMx_CH1_AF GPIO_AF_TIM2 #define TIMx_CH2_GPIO_PORT GPIOA #define TIMx_CH2_GPIO_PIN GPIO_Pin_1 #define TIMx_CH2_GPIO_CLK RCC_AHB1Periph_GPIOA #define TIMx_CH2_SOURCE GPIO_PinSource1 #define TIMx_CH2_AF GPIO_AF_TIM2 // 初始化定时器 void TIM_Config(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_ICInitTypeDef TIM_ICInitStructure; GPIO_InitTypeDef GPIO_InitStructure; NVIC_InitTypeDef NVIC_InitStructure; // 使能定时器和GPIO时钟 RCC_APB1PeriphClockCmd(TIMx_CLK, ENABLE); RCC_AHB1PeriphClockCmd(TIMx_CH1_GPIO_CLK | TIMx_CH2_GPIO_CLK, ENABLE); // 配置GPIO为TIMx通道1和通道2 GPIO_InitStructure.GPIO_Pin = TIMx_CH1_GPIO_PIN | TIMx_CH2_GPIO_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_100MHz; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP ; GPIO_Init(GPIOA, &GPIO_InitStructure); // 将GPIO引脚映射到TIMx通道1和通道2上 GPIO_PinAFConfig(TIMx_CH1_GPIO_PORT, TIMx_CH1_SOURCE, TIMx_CH1_AF); GPIO_PinAFConfig(TIMx_CH2_GPIO_PORT, TIMx_CH2_SOURCE, TIMx_CH2_AF); // 配置定时器为输入捕获模式 TIM_TimeBaseStructure.TIM_Period = 0xFFFF; TIM_TimeBaseStructure.TIM_Prescaler = 0; TIM_TimeBaseStructure.TIM_ClockDivision = 0; TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; TIM_TimeBaseInit(TIMx, &TIM_TimeBaseStructure); TIM_ICInitStructure.TIM_Channel = TIM_Channel_1; TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; TIM_ICInitStructure.TIM_ICFilter = 0x0; TIM_ICInit(TIMx, &TIM_ICInitStructure); TIM_ICInitStructure.TIM_Channel = TIM_Channel_2; TIM_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; TIM_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; TIM_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; TIM_ICInitStructure.TIM_ICFilter = 0x0; TIM_ICInit(TIMx, &TIM_ICInitStructure); // 使能定时器中断 NVIC_InitStructure.NVIC_IRQChannel = TIMx_IRQn; NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); // 启动定时器 TIM_Cmd(TIMx, ENABLE); // 启用定时器的捕获中断 TIM_ITConfig(TIMx, TIM_IT_CC1 | TIM_IT_CC2, ENABLE); } // 定时器中断处理函数 void TIMx_IRQHandler(void) { if (TIM_GetITStatus(TIMx, TIM_IT_CC1) != RESET) { // 处理A相信号 TIM_ClearITPendingBit(TIMx, TIM_IT_CC1); } else if (TIM_GetITStatus(TIMx, TIM_IT_CC2) != RESET) { // 处理B相信号 TIM_ClearITPendingBit(TIMx, TIM_IT_CC2); } } ``` 3. 通过读取编码器信号,计算电机的转速和位置 当定时器捕获到编码器信号的上升沿或下降沿时,您需要更新电机的位置和速度。在处理A相信号时,如果B相信号也发生了变化,则电机向前转动;如果B相信号没有发生变化,则电机向后转动。在处理B相信号时,您可以使用相同的逻辑来确定电机的方向。下面是一个示例实现: ```c // 定义编码器参数 #define ENCODER_RESOLUTION 1000.0f // 编码器分辨率 #define WHEEL_DIAMETER 0.1f // 轮子直径(单位:米) #define GEAR_RATIO 100.0f // 减速比 #define PI 3.1415926 // 定义电机状态 typedef struct { uint32_t position; // 电机的位置(单位:脉冲) float speed; // 电机的速度(单位:转/秒) } motor_t; motor_t motor; // 处理A相信号中断 void handle_encoder_A_interrupt(void) { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) == GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1)) { // 编码器向前转动 motor.position++; } else { // 编码器向后转动 motor.position--; } } // 处理B相信号中断 void handle_encoder_B_interrupt(void) { if (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1) == GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0)) { // 编码器向前转动 motor.position++; } else { // 编码器向后转动 motor.position--; } } // 计算电机的速度和位置 void calculate_motor_speed_and_position(void) { static uint32_t last_position = 0; static uint32_t last_time = 0; // 计算电机的位置 uint32_t current_position = motor.position; float delta_position = current_position - last_position; last_position = current_position; // 计算电机的速度 uint32_t current_time = TIM_GetCounter(TIM2); float delta_time = current_time - last_time; last_time = current_time; float delta_angle = delta_position / ENCODER_RESOLUTION * 2 * PI / GEAR_RATIO; float delta_distance = delta_angle * WHEEL_DIAMETER / 2; motor.speed = delta_distance / delta_time; } ``` 4. 使用电机的转速和位置信息,计算电机的线速度和加速度 使用电机的速度和位置信息,可以计算电机的线速度和加速度。您需要将电机的速度转换为线速度,然后使用两个连续的速度值来计算电机的加速度。下面是一个示例实现: ```c // 计算电机的线速度和加速度 void calculate_motor_linear_speed_and_acceleration(float *linear_speed, float *acceleration) { static float last_speed = 0; static uint32_t last_time = 0; // 计算电机的线速度 *linear_speed = motor.speed * WHEEL_DIAMETER / 2; // 计算电机的加速度 uint32_t current_time = TIM_GetCounter(TIM2); float delta_time = current_time - last_time; last_time = current_time; *acceleration = (*linear_speed - last_speed) / delta_time; last_speed = *linear_speed; } ``` 5. 将电机的线速度和加速度转换为距离 最后,您可以使用电机的线速度和加速度来计算电机的距离。下面是一个示例实现: ```c // 计算电机移动的距离 void calculate_motor_distance(float *distance) { static float last_speed = 0; static uint32_t last_time = 0; // 计算电机的线速度和加速度 float linear_speed, acceleration; calculate_motor_linear_speed_and_acceleration(&linear_speed, &acceleration); // 计算电机移动的距离 uint32_t current_time = TIM_GetCounter(TIM2); float delta_time = current_time - last_time; last_time = current_time; *distance += (last_speed + linear_speed) / 2 * delta_time; last_speed = linear_speed; } ``` 以上是一个简单的示例,可以帮助您开始使用STM32C8T6和JGB37-520编码器电机进行测距。但是,请注意,您需要进行更多的调试和测试,以确保代码的正确性和可靠性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值