当前位置:首页 > 单片机 > 单片机
[导读]STM32正交编码器驱动,引入(突变)带进位的位置环和速度环 http://bbs.elecfans.com/jishu_484159_1_1.html (出处: 中国电子技术论坛)#include "stm32f10x.h"#include "stm32f10x_encoder.h"#include "sys.h"#inclu

STM32正交编码器驱动,引入(突变)带进位的位置环和速度环 http://bbs.elecfans.com/jishu_484159_1_1.html (出处: 中国电子技术论坛)



#include "stm32f10x.h"
#include "stm32f10x_encoder.h"
#include "sys.h"
#include "usart.h"
#include "led.h"




#define COUNTER_RESET(u16)0
#define ICx_FILTER(u8) 0 // 6<-> 670nsec

#define TIMx_PRE_EMPTION_PRIORITY 1
#define TIMx_SUB_PRIORITY 0

//#define MAX_COUNT ENCODER_TIM_PERIOD/2





volatile s16 Right_hEncoder_Timer_Overflow; //编码器计数值溢出标志
volatile s16 Right_hEncoder_Timer_Overflow_High;//编码器计数值溢出标志高位
volatile s16 Right_hRot_Speed;//当前的速度
volatile s16 Right_hRot_Acceleration;//当前的加速度

s32 Right_CurrentCount = 0;//编码器当前的总计数值
s32 Right_CurrentCount_high = 0;//编码器当前的总计数值高位计算公式,注意结果可能超过32位的表示范围Left_CurrentCount_high*S32_MAX+Left_CurrentCount

volatile s16 Left_hEncoder_Timer_Overflow;
volatile s16 Left_hEncoder_Timer_Overflow_High;
volatile s16 Left_hRot_Speed;
volatile s16 Left_hRot_Acceleration;

s32 Left_CurrentCount = 0;
s32 Left_CurrentCount_high = 0;

void ENC_Right_Init(void)
{
TIM_TimeBaseInitTypeDefTIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;


GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;//A7接靠近红色的编码器输入一端
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);


NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = TIMx_PRE_EMPTION_PRIORITY;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = TIMx_SUB_PRIORITY;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);


TIM_DeInit(ENCODER_Right_TIMER);
TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);

TIM_TimeBaseStructure.TIM_Prescaler = 0x0;// No prescaling
TIM_TimeBaseStructure.TIM_Period = (4*ENCODER_PPR)-1;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(ENCODER_Right_TIMER, &TIM_TimeBaseStructure);
//TIMx_SMCR 寄存器中的 SMS=011 = TIM_EncoderMode_TI12;
TIM_EncoderInterfaceConfig(ENCODER_Right_TIMER, TIM_EncoderMode_TI12,
TIM_ICPolarity_Rising, TIM_ICPolarity_Rising); //ICx 不反相输入捕获在上升沿
TIM_ICStructInit(&TIM_ICInitStructure); //TIM_Channel_1 TIM_ICPSC_DIV1 TIM_ICPolarity_Rising
TIM_ICInitStructure.TIM_ICFilter = ICx_FILTER;//输入滤波 0110:采样频率fSAMPLING=fDTS/4, N=6
TIM_ICInit(ENCODER_Right_TIMER, &TIM_ICInitStructure);

// Clear all pending interrupts
TIM_ClearFlag(ENCODER_Right_TIMER, TIM_FLAG_Update);
TIM_ITConfig(ENCODER_Right_TIMER, TIM_IT_Update, ENABLE);
//Reset counter
ENCODER_Right_TIMER->CNT = COUNTER_RESET;



TIM_Cmd(ENCODER_Right_TIMER, ENABLE);
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
Right_hEncoder_Timer_Overflow = 0;
Right_hEncoder_Timer_Overflow_High = 0;
Right_hRot_Speed = 0;
Right_ENC_Get_Electrical_Angle(); //第一次不要这个数据!!lastCount
}


void ENC_Left_Init(void)
{
TIM_TimeBaseInitTypeDefTIM_TimeBaseStructure;
TIM_ICInitTypeDef TIM_ICInitStructure;


GPIO_InitTypeDef GPIO_InitStructure;
NVIC_InitTypeDef NVIC_InitStructure;


RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);

GPIO_StructInit(&GPIO_InitStructure);

GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1;//A7接靠近红色的编码器输入一端
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);


NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = TIMx_PRE_EMPTION_PRIORITY;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = TIMx_SUB_PRIORITY+1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);


TIM_DeInit(ENCODER_Left_TIMER);
TIM_TimeBaseStructInit(&TIM_TimeBaseStructure);

TIM_TimeBaseStructure.TIM_Prescaler = 0x0;// No prescaling
TIM_TimeBaseStructure.TIM_Period = (4*ENCODER_PPR)-1;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(ENCODER_Left_TIMER, &TIM_TimeBaseStructure);
//TIMx_SMCR 寄存器中的 SMS=011 = TIM_EncoderMode_TI12;
TIM_EncoderInterfaceConfig(ENCODER_Left_TIMER, TIM_EncoderMode_TI12,
TIM_ICPolarity_Rising, TIM_ICPolarity_Rising); //ICx 不反相输入捕获在上升沿
TIM_ICStructInit(&TIM_ICInitStructure); //TIM_Channel_1 TIM_ICPSC_DIV1 TIM_ICPolarity_Rising
TIM_ICInitStructure.TIM_ICFilter = ICx_FILTER;//输入滤波 0110:采样频率fSAMPLING=fDTS/4, N=0
TIM_ICInit(ENCODER_Left_TIMER, &TIM_ICInitStructure);

// Clear all pending interrupts
TIM_ClearFlag(ENCODER_Left_TIMER, TIM_FLAG_Update);
TIM_ITConfig(ENCODER_Left_TIMER, TIM_IT_Update, ENABLE);
//Reset counter
ENCODER_Left_TIMER->CNT = COUNTER_RESET;



TIM_Cmd(ENCODER_Left_TIMER, ENABLE);
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
__nop();
Left_hEncoder_Timer_Overflow = 0;
Left_hEncoder_Timer_Overflow_High = 0;
Left_hRot_Speed = 0;
Left_ENC_Get_Electrical_Angle(); //第一次不要这个数据!!lastCount
}


s16 Right_ENC_Get_Electrical_Angle(void)
{
static u16lastCount = 0;
u16 curCount ;
s16 dAngle ;
static s16 Last_Overflow = 0;


curCount = ENCODER_Right_TIMER->CNT;
dAngle = curCount - lastCount;

//ensure |Right_hEncoder_Timer_Overflow - Last_Overflow|<=1
if(Right_hEncoder_Timer_Overflow != Last_Overflow)
{

if ( (ENCODER_Right_TIMER->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down)//反转
{// encoder timer down-counting
dAngle -= ENCODER_TIM_PERIOD;

}
else
{//encoder timer up-counting

dAngle += ENCODER_TIM_PERIOD;

}

}

lastCount = curCount;
Last_Overflow = Right_hEncoder_Timer_Overflow;
return (s16)dAngle;

}


s16 Left_ENC_Get_Electrical_Angle(void)
{
static u16lastCount = 0;
u16 curCount ;
s16 dAngle ;
static s16 Last_Overflow = 0;


curCount = ENCODER_Left_TIMER->CNT;
dAngle = curCount - lastCount;

//ensure |Left_hEncoder_Timer_Overflow - Last_Overflow|<=1
if(Left_hEncoder_Timer_Overflow != Last_Overflow)
{

if ( (ENCODER_Left_TIMER->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down)//反转
{// encoder timer down-counting
dAngle -= ENCODER_TIM_PERIOD;

}
else
{//encoder timer up-counting

dAngle += ENCODER_TIM_PERIOD;

}

}

lastCount = curCount;
Last_Overflow = Left_hEncoder_Timer_Overflow;
return (s16)dAngle;

}

void TIM2_IRQHandler(void)
{

TIM_ClearFlag(ENCODER_Left_TIMER, TIM_FLAG_Update);

if ((Left_hEncoder_Timer_Overflow != S16_MAX )&&(Left_hEncoder_Timer_Overflow != S16_MIN ))
{
if ( (ENCODER_Left_TIMER->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down)//反转
{// encoder timer down-counting
Left_hEncoder_Timer_Overflow--;
}
else
{//encoder timer up-counting
Left_hEncoder_Timer_Overflow++;
}

}
if(Left_hEncoder_Timer_Overflow == S16_MAX)
{
Left_hEncoder_Timer_Overflow = 0;
Left_hEncoder_Timer_Overflow_High++;
}
if(Left_hEncoder_Timer_Overflow == S16_MIN )
{
Left_hEncoder_Timer_Overflow = 0;
Left_hEncoder_Timer_Overflow_High--;

}

}




void TIM3_IRQHandler(void)
{

TIM_ClearFlag(ENCODER_Right_TIMER, TIM_FLAG_Update);

if ((Right_hEncoder_Timer_Overflow != S16_MAX )&&(Right_hEncoder_Timer_Overflow != S16_MIN ))
{
if ( (ENCODER_Right_TIMER->CR1 & TIM_CounterMode_Down) == TIM_CounterMode_Down)//反转
{// encoder timer down-counting
Right_hEncoder_Timer_Overflow--;

}
else
{//encoder timer up-counting
Right_hEncoder_Timer_Overflow++;
}

}
if(Right_hEncoder_Timer_Overflow == S16_MAX)
{
Right_hEncoder_Timer_Overflow = 0;
Right_hEncoder_Timer_Overflow_High++;
}
if(Right_hEncoder_Timer_Overflow == S16_MIN )
{
Right_hEncoder_Timer_Overflow = 0;
Right_hEncoder_Timer_Overflow_High--;

}

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭
关闭