11、STM32—编码器接口(代码示例)

一、编码器接口简介

 (1)Encoder Interface 编码器接口

       <1> 编码器接口可接收增量(正交)编码器的信号,根据编码器旋转产生的正交信号脉冲, 自动控制CNT自增或自减 ,从而指示 编码器的位置 旋转方向 旋转速度
        <2>每个 高级定时器和通用定时器 都拥有 1个编码器接口
       <3> 两个输入引脚借用了输入捕获的通道 1和通道2

  (2)正交编码器

二、编码器接口基本结构框图

三、编码器接口测速代码

        

#include "stm32f10x.h"                  // Device header
#include "OLED.h"

 uint16_t speed=0;
void TimerInit(void);
void EncoderInit(void);
int16_t EncoderGetCount(void);

int main(void)
{
	TimerInit();
  OLED_Init();
  EncoderInit();
	while(1)
	{

		OLED_ShowNum(1,7,speed,5);
		
	}

}


/***************************************************************
  *  @brief void TimerInit(void)   
  *  @param void       
  *  @note  初始化中断引脚  
  *  @Sample TimerInit();  
 **************************************************************/
void TimerInit(void)
{
	TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
  RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2,ENABLE);
	NVIC_InitTypeDef NVIC_InitStruct;
	
  TIM_InternalClockConfig(TIM2);//选用内部时钟
	
	TIM_TimeBaseInitStruct.TIM_ClockDivision=TIM_CKD_DIV1;//指定时钟分频
	TIM_TimeBaseInitStruct.TIM_CounterMode=TIM_CounterMode_Up;//计数器模式
	TIM_TimeBaseInitStruct.TIM_Period=10000-1;//周期ARR自动重装载值
	TIM_TimeBaseInitStruct.TIM_Prescaler=7200-1;//PSC预分频器的值
	TIM_TimeBaseInitStruct.TIM_RepetitionCounter=0;//重复计数器的值
	//定时频率=72M/(PSC+1)*(ARR+1)
	//定时时间=1/[72/(PSC+1)]*(ARR+1)
  TIM_TimeBaseInit(TIM2,&TIM_TimeBaseInitStruct);
	
	TIM_ITConfig(TIM2,TIM_IT_Update,ENABLE);//使能中断
  
	NVIC_InitStruct.NVIC_IRQChannel=TIM2_IRQn;//中断通道
	NVIC_InitStruct.NVIC_IRQChannelCmd=ENABLE;//
	NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority=2;//
	NVIC_InitStruct.NVIC_IRQChannelSubPriority=1;//
	
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//选择分组2
	NVIC_Init(&NVIC_InitStruct);
	
  TIM_Cmd(TIM2,ENABLE);//启动定时器
	
}

/***************************************************************
  *  @brief   void TIM2_IRQHandler()  
  *  @param   void     
  *  @note    定时器中断函数
  *  @Sample  无 
 **************************************************************/
void TIM2_IRQHandler()
{
	if( TIM_GetITStatus(TIM2,TIM_IT_Update)==SET)//判断中断标志位是否被置一
	{
	speed=EncoderGetCount();
	TIM_ClearITPendingBit(TIM2,TIM_IT_Update);
	}
	
}

/***************************************************************
  *  @brief  void EncoderInit(void)   
  *  @param  void      
  *  @note   编码器模式初始化 
  *  @Sample EncoderInit();  
 **************************************************************/
void EncoderInit(void)
{
 //1、开始时钟
 TIM_ICInitTypeDef TIM_ICInitStruct;
 GPIO_InitTypeDef GPIO_InitStruct;
 TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
 RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3,ENABLE);
 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);
//2、配置GPIO引脚
 GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IPU;
 GPIO_InitStruct.GPIO_Pin=GPIO_Pin_6|GPIO_Pin_7;
 GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;
 GPIO_Init(GPIOA,&GPIO_InitStruct);
 
 TIM_TimeBaseInitStruct.TIM_ClockDivision=TIM_CKD_DIV1;//指定时钟分频
 TIM_TimeBaseInitStruct.TIM_CounterMode=TIM_CounterMode_Up;//计数器模式
 TIM_TimeBaseInitStruct.TIM_Period=65536-1;//周期ARR自动重装载值
 TIM_TimeBaseInitStruct.TIM_Prescaler=72-1;//PSC预分频器的值 决定了测周法的标准频率fc
 TIM_TimeBaseInitStruct.TIM_RepetitionCounter=0;//重复计数器的值
 //定时频率=72M/(PSC+1)*(ARR+1)
 //定时时间=1/[72/(PSC+1)]*(ARR+1)
 TIM_TimeBaseInit(TIM3,&TIM_TimeBaseInitStruct);
//3、配置输入捕获模式
 TIM_ICStructInit(&TIM_ICInitStruct);
 TIM_ICInitStruct.TIM_Channel=TIM_Channel_1;//选择通道
 TIM_ICInitStruct.TIM_ICFilter=0xf;//设置输入捕获的滤波器
 TIM_ICInitStruct.TIM_ICPolarity=TIM_ICPolarity_Rising;//极性(触发方式)高低电平不反转
 TIM_ICInit(TIM3,&TIM_ICInitStruct);
	
 TIM_ICInitStruct.TIM_Channel=TIM_Channel_2;//选择通道
 TIM_ICInitStruct.TIM_ICFilter=0xf;//设置输入捕获的滤波器
 TIM_ICInitStruct.TIM_ICPolarity=TIM_ICPolarity_Rising;//极性(触发方式)高低电平不反转
 TIM_ICInit(TIM3,&TIM_ICInitStruct);
	
 //4、设置编码器模式
 TIM_EncoderInterfaceConfig(TIM3,TIM_EncoderMode_TI12,TIM_ICPolarity_Rising,TIM_ICPolarity_Rising);

 TIM_Cmd(TIM3,ENABLE);
}

/***************************************************************
  *  @brief int16_t EncoderGetCount(void)    
  *  @param void       
  *  @note  获取转动速度  
  *  @Sample  EncoderGetCount( ); 
 **************************************************************/
int16_t EncoderGetCount(void)
{
	int16_t Tem;
	Tem=TIM_GetCounter(TIM3);
	TIM_SetCounter(TIM3,0);
	return Tem;
}

STM32 编码器是一种用于检测旋转轴的位置或速度的设备,在电子设备尤其是工业控制、机器人技术等领域广泛应用。为了处理编码器数据并利用 STM32 微控制器(一款广泛应用于嵌入式系统的高性能微处理器),您需要编写特定的代码。 ### STM32 编码器基本工作原理 编码器通常包含一组对称排列的脉冲输出信号,当轴旋转时,这些信号会交替变化。例如,对于常见的增量式编码器,每个完整的圆周运动会产生一定数量的脉冲序列。这些信号由编码器连接到 STM32 的模拟输入通道上,并通过ADC(模数转换器)转换成数字信号。 ### STM32 编码器代码示例: 假设我们使用了STM32F10X系列芯片,下面是一个简单的示例代码片段,用于读取来自一个单相增量型编码器的脉冲,并计数其位置信息。 ```c #include "stm32f1xx_hal.h" // 引入HAL库头文件 // 定义中断服务函数 - 计算每圈转动次数的中断 void EncoderISR(void) { __disable_irq(); // 关闭中断 int count = digitalRead(PIN_A) + (digitalRead(PIN_B) << 1); // PIN_A 和 PIN_B 分别连接编码器的两个通道 if (count >= COUNT_THRESHOLD) { // 每经过一定次数的计数视为一圈完成 encoderPosition++; if(encoderPosition > MAX_POSITION){ encoderPosition = 0; // 保持在有效范围内 } // 可以在这里添加其他操作,比如更新显示、发送数据等 HAL_NVIC_SetPriority(Encoder_IRQn, Encoder_IRQ_PRIORITY, Encoder_IRQ_FLAG); HAL_NVIC_ClearPendingIRQ(Encoder_IRQn); __enable_irq(); // 开启中断 } else { __enable_irq(); // 直接开启中断,不需要等待下一圈 } } int main() { GPIO_InitTypeDef GPIO_InitStruct; // 初始化GPIO和ADC配置 ADC_ChannelConfTypeDef sConfig; // ... 省略初始化代码 ... /* 配置编码器的A/B信号线 */ GPIO_InitStruct.Pin = PIN_A | PIN_B; GPIO_InitStruct.Mode = GPIO_MODE_IN_IT_FALLING | GPIO_MODE_IN; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(GPIOA, &GPIO_InitStruct); HAL_ADC_Init(&hadc1); sConfig.Channel = ADC_CHANNEL_1; // 使用ADC通道1作为编码器的输入端口 sConfig.Rank = 1; // 设置通道优先级 sConfig.SamplingTime = ADC_SAMPLETIME_7CYCLES_5; // 设置采样时间 sConfig.Offset = 0x40; // 根据实际应用调整偏移量 sConfig.GainRange = ADC_GANC_RANGE_V12P8; // 设置增益范围 sConfig.DataAlign = ADC_DATAALIGN_RIGHT; // 数据对齐方式 sConfig.NbrOfConversion = 1; // 单次转换 HAL_ADC_ConfigChannel(&hadc1, &sConfig); HAL_ADC_Start_DMA(&hadc1, (uint32_t*)adcDataBuffer, adcBufferSize); /* 注册中断事件 */ HAL_NVIC_Init(); HAL_NVIC_SetPriority(ADC_IRQn, ADC_IRQ_PRIORITY, ADC_IRQ_FLAG); HAL_NVIC_EnableIRQ(ADC_IRQn); HAL_NVIC_SetPriority(SysTick_IRQn, SysTick_IRQ_PRIORITY, SysTick_IRQ_FLAG); HAL_NVIC_EnableIRQ(SysTick_IRQn); HAL_NVIC_SetPriority(Encoder_IRQn, Encoder_IRQ_PRIORITY, Encoder_IRQ_FLAG); HAL_NVIC_EnableIRQ(Encoder_IRQn); while (1) { // 主循环可以在此处添加更多功能... } return 0; } ``` ### 注意事项: - **编码器的配置**:确保正确设置编码器的脉冲计数规则(例如,正反转如何影响计数方向)以及确定一“圈”内产生的脉冲数量。 - **中断管理**:编码器中断应当高效地处理,避免不必要的延迟影响系统性能。 - **ADC配置**:根据编码器的具体规格调整ADC配置,包括采样率、通道选择等。 ###
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值