使用STM32F103采集天地飞遥控的信号

2019-08-17 09:52发布

最近在开始了新一轮的四轴飞行器制作,使用的是机架模型而不是自己的PCB做出来的机架,遥控也不是以前的自制蓝牙遥控而是天地飞的专业遥控。谨以此文作为开始本次旅程的一个起点。 使用遥控控制电机的转动,原理上是通过遥控改变MCU输出PWM的高低电平时间来改变电机的转速,本次设计中使用的是朗宇2212的无刷直流电机,1ms高电平时间为电机停止,2ms高电平时间为电机全速转动。相信这个对于做四轴的同学来说都已经有所了解,知道这个之后,**我觉得应该使用STM32进行PWM输入捕捉对遥控信号进行采样来得到控制输入电机的量**(思维误区),然后STM32根据输入信号的变化,对应的输出去控制电机的转速来控制四轴的平衡。 到了这一步,我们该进行遥控信号的输入采集了,由于要采集的是一个PWM信号,第一时间想到的是使用PWM输入捕捉功能,话不多说,数据手册看了看,库函数手册看了看,发现有问题了,数据手册上明明显示着只有CH1,CH2才能进行PWM输入捕捉。知道这个消息之后,当时的我内心是崩溃的!!!之后强行上各种论坛搜索前辈的经验,发现并没有想我这么是在的娃问这种问题或是前辈给出着相关的解释,知道这个消息后,再度崩溃!!!各种专业QQ群问遍之后,我开始放弃去问问题了,想了想,这个遥控用的人挺多的,应该有人有解决的方案,所以应该有合适的输入捕捉方式。 回到上面的思维误区来,我想的是采集PWM信号。但是前面说过的一个电机控制问题,我需要的是在1ms至2ms范围内的高电平来控制一个电机转动而已,我需要的只是高电平的时间啊!**大概理了理想法之后,觉得这个应该是可以用普通的输入捕捉来实现遥控信号的采集的。**(之前做的PWM输入采集这时候就可以用得上了,我已经知道了遥控出来的PWM信号所有信息,这样对电机的配置就可以提供一个适当的参考),话不多说了,上一个关于输入捕捉4lu高电平信号的栗子。 定时器4初始化:[/mw_shl_code] void TIM4_Cap_Init(u16 arr, u16 psc) 

GPIO_InitTypeDef GPIO_InitStructure; 
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; 
NVIC_InitTypeDef NVIC_InitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM4, ENABLE); //使能TIM4时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能GPIOB时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8 | GPIO_Pin_9; //PB6,7,8,9 清除之前设置 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PB6,7,8,9 输入 GPIO_Init(GPIOB, &GPIO_InitStructure); GPIO_ResetBits(GPIOB, GPIO_Pin_6 | GPIO_Pin_7 | GPIO_Pin_8 | GPIO_Pin_9);//PB6,7,8,9 下拉 //初始化定时器4 TIM4 TIM_TimeBaseStructure.TIM_Period = arr; //设定计数器自动重装值 TIM_TimeBaseStructure.TIM_Prescaler = psc; //预分频器 TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式 TIM_TimeBaseInit(TIM4, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位 //初始化TIM4输入捕获参数 通道1 TIM4_ICInitStructure.TIM_Channel = TIM_Channel_1; //CC1S=01 选择输入端 IC1映射到TI1上 TIM4_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿捕获 TIM4_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上 TIM4_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //配置输入分频,不分频 TIM4_ICInitStructure.TIM_ICFilter = 0x00; //IC1F=0000 配置输入滤波器 不滤波 TIM_ICInit(TIM4, &TIM4_ICInitStructure); //初始化TIM4输入捕获参数 通道2 TIM4_ICInitStructure.TIM_Channel = TIM_Channel_2; //CC1S=01 选择输入端 IC1映射到TI1上 TIM4_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿捕获 TIM4_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上 TIM4_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //配置输入分频,不分频 TIM4_ICInitStructure.TIM_ICFilter = 0x00; //IC1F=0000 配置输入滤波器 不滤波 TIM_ICInit(TIM4, &TIM4_ICInitStructure); //初始化TIM4输入捕获参数 通道3 TIM4_ICInitStructure.TIM_Channel = TIM_Channel_3; //CC1S=01 选择输入端 IC1映射到TI1上 TIM4_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿捕获 TIM4_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上 TIM4_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //配置输入分频,不分频 TIM4_ICInitStructure.TIM_ICFilter = 0x00; //IC1F=0000 配置输入滤波器 不滤波 TIM_ICInit(TIM4, &TIM4_ICInitStructure); //初始化TIM4输入捕获参数 通道4 TIM4_ICInitStructure.TIM_Channel = TIM_Channel_4; //CC1S=01 选择输入端 IC1映射到TI1上 TIM4_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising; //上升沿捕获 TIM4_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上 TIM4_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1; //配置输入分频,不分频 TIM4_ICInitStructure.TIM_ICFilter = 0x00; //IC1F=0000 配置输入滤波器 不滤波 TIM_ICInit(TIM4, &TIM4_ICInitStructure); //中断分组初始化 NVIC_InitStructure.NVIC_IRQChannel = TIM4_IRQn; //TIM4中断 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级1级 //NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; //从优先级0级 NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能 NVIC_Init(&NVIC_InitStructure); //根据NVIC_InitStruct中指定的参数初始化外设NVIC寄存器 TIM_ITConfig(TIM4, TIM_IT_CC1 | TIM_IT_CC2 | TIM_IT_CC3 | TIM_IT_CC4, ENABLE); //不允许更新中断,允许CC1IE,CC2IE,CC3IE,CC4IE捕获中断 TIM_Cmd(TIM4, ENABLE); //使能定时器4[/mw_shl_code] } 定时器4的中断: 
//定时器4中断服务程序 
void TIM4_IRQHandler(void) 

if ((TIM4CH1_CAPTURE_STA & 0X80) == 0) //还未成功捕获 

if (TIM_GetITStatus(TIM4, TIM_IT_CC1) != RESET) //捕获1发生捕获事件 

TIM_ClearITPendingBit(TIM4, TIM_IT_CC1); //清除中断标志位 
if (TIM4CH1_CAPTURE_STA & 0X40) //捕获到一个下降沿 

TIM4CH1_CAPTURE_DOWNVAL = TIM_GetCapture1(TIM4);//记录下此时的定时器计数值 
if (TIM4CH1_CAPTURE_DOWNVAL < TIM4CH1_CAPTURE_UPVAL) 

tim4_T1 = 65535; 

else 
tim4_T1 = 0; 
tempup1 = TIM4CH1_CAPTURE_DOWNVAL - TIM4CH1_CAPTURE_UPVAL 
+ tim4_T1; //得到总的高电平的时间 
pwmout1 = tempup1; //总的高电平的时间 
TIM4CH1_CAPTURE_STA = 0; //捕获标志位清零 
TIM_OC1PolarityConfig(TIM4, TIM_ICPolarity_Rising); //设置为上升沿捕获 

else //发生捕获时间但不是下降沿,第一次捕获到上升沿,记录此时的定时器计数值 

TIM4CH1_CAPTURE_UPVAL = TIM_GetCapture1(TIM4); //获取上升沿数据 
TIM4CH1_CAPTURE_STA |= 0X40; //标记已捕获到上升沿 
TIM_OC1PolarityConfig(TIM4, TIM_ICPolarity_Falling);//设置为下降沿捕获 


} if ((TIM4CH2_CAPTURE_STA & 0X80) == 0) //还未成功捕获 { if (TIM_GetITStatus(TIM4, TIM_IT_CC2) != RESET) //捕获2发生捕获事件 { TIM_ClearITPendingBit(TIM4, TIM_IT_CC2); //清除中断标志位 if (TIM4CH2_CAPTURE_STA & 0X40) //捕获到一个下降沿 { TIM4CH2_CAPTURE_DOWNVAL = TIM_GetCapture2(TIM4);//记录下此时的定时器计数值 if (TIM4CH2_CAPTURE_DOWNVAL < TIM4CH2_CAPTURE_UPVAL) { tim4_T2 = 65535; } else tim4_T2 = 0; tempup2 = TIM4CH2_CAPTURE_DOWNVAL - TIM4CH2_CAPTURE_UPVAL + tim4_T2; //得到总的高电平的时间 pwmout2 = tempup2; //总的高电平的时间 TIM4CH2_CAPTURE_STA = 0; //捕获标志位清零 TIM_OC2PolarityConfig(TIM4, TIM_ICPolarity_Rising); //设置为上升沿捕获 } else //发生捕获时间但不是下降沿,第一次捕获到上升沿,记录此时的定时器计数值 { TIM4CH2_CAPTURE_UPVAL = TIM_GetCapture2(TIM4); //获取上升沿数据 TIM4CH2_CAPTURE_STA |= 0X40; //标记已捕获到上升沿 TIM_OC2PolarityConfig(TIM4, TIM_ICPolarity_Falling);//设置为下降沿捕获 } } } if ((TIM4CH3_CAPTURE_STA & 0X80) == 0) //还未成功捕获 { if (TIM_GetITStatus(TIM4, TIM_IT_CC3) != RESET) //捕获3发生捕获事件 { TIM_ClearITPendingBit(TIM4, TIM_IT_CC3); //清除中断标志位 if (TIM4CH3_CAPTURE_STA & 0X40) //捕获到一个下降沿 { TIM4CH3_CAPTURE_DOWNVAL = TIM_GetCapture3(TIM4);//记录下此时的定时器计数值 if (TIM4CH3_CAPTURE_DOWNVAL < TIM4CH3_CAPTURE_UPVAL) { tim4_T3 = 65535; } else tim4_T3 = 0; tempup3 = TIM4CH3_CAPTURE_DOWNVAL - TIM4CH3_CAPTURE_UPVAL + tim4_T3; //得到总的高电平的时间 pwmout3 = tempup3; //总的高电平的时间 TIM4CH3_CAPTURE_STA = 0; //捕获标志位清零 TIM_OC3PolarityConfig(TIM4, TIM_ICPolarity_Rising); //设置为上升沿捕获 } else //发生捕获时间但不是下降沿,第一次捕获到上升沿,记录此时的定时器计数值 { TIM4CH3_CAPTURE_UPVAL = TIM_GetCapture3(TIM4); //获取上升沿数据 TIM4CH3_CAPTURE_STA |= 0X40; //标记已捕获到上升沿 TIM_OC3PolarityConfig(TIM4, TIM_ICPolarity_Falling);//设置为下降沿捕获 } } } if ((TIM4CH4_CAPTURE_STA & 0X80) == 0) //还未成功捕获 { if (TIM_GetITStatus(TIM4, TIM_IT_CC4) != RESET) //捕获4发生捕获事件 { TIM_ClearITPendingBit(TIM4, TIM_IT_CC4); //清除中断标志位 if (TIM4CH4_CAPTURE_STA & 0X40) //捕获到一个下降沿 { TIM4CH4_CAPTURE_DOWNVAL = TIM_GetCapture4(TIM4);//记录下此时的定时器计数值 if (TIM4CH4_CAPTURE_DOWNVAL < TIM4CH4_CAPTURE_UPVAL) { tim4_T4 = 65535; } else tim4_T4 = 0; tempup4 = TIM4CH4_CAPTURE_DOWNVAL - TIM4CH4_CAPTURE_UPVAL + tim4_T4; //得到总的高电平的时间 pwmout4 = tempup4; //总的高电平的时间 TIM4CH4_CAPTURE_STA = 0; //捕获标志位清零 TIM_OC4PolarityConfig(TIM4, TIM_ICPolarity_Rising); //设置为上升沿捕获 } else //发生捕获时间但不是下降沿,第一次捕获到上升沿,记录此时的定时器计数值 { TIM4CH4_CAPTURE_UPVAL = TIM_GetCapture4(TIM4); //获取上升沿数据 TIM4CH4_CAPTURE_STA |= 0X40; //标记已捕获到上升沿 TIM_OC4PolarityConfig(TIM4, TIM_ICPolarity_Falling);//设置为下降沿捕获 } } }[/mw_shl_code] } 配置到这一步的话已经可以了,上一个图。 
这里写图片描述
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
8条回答
正点原子
1楼-- · 2019-08-17 13:35
谢谢分享...
Alone_
2楼-- · 2019-08-17 14:18
pwm频率是多少啊
CCS
3楼-- · 2019-08-17 20:09
谢谢分享,能否问个问题:TIM4CH4_CAPTURE_UPVAL这个变量是代表捕获到上升沿时的计数值吧?同时TIM4CH4_CAPTURE_UPVAL的初始赋值是多少?就还有这点不太明白,谢谢了!
路口农夫
4楼-- · 2019-08-18 00:37
 精彩回答 2  元偷偷看……
1606954330
5楼-- · 2019-08-18 02:49
为什么不允许更新中断呢?
1606954330
6楼-- · 2019-08-18 06:10
GPIO_Mode_IPD;就已经下拉了 为毛还要在输出低电平

一周热门 更多>