单片机学习小组
直播中

文妮

7年用户 246经验值
私信 关注

mwc基于stm32四轴飞行器源码

基于STM32四轴飞行器源码有c语言源码,应用keil5 即可得到最终结果

单片机源程序如下:
  • /**********************************************************************************
  • * 文件名  :incap.c
  • * 描述    :四通道输入捕获
  • * 实验平台:四轴飞行器STM3主控板
  • * 库版本  :ST3.5
  • * 作者    :
  • * 时间    :
  • **********************************************************************************/
  • #include "PPM.h"
  • //定时器5通道1输入捕获配置
  • u8  tiM5CH1_CAPTURE_STA=0;        //通道1输入捕获状态
  • u16        TIM5CH1_CAPTURE_VAL;        //通道1输入捕获值
  • u8  TIM5CH2_CAPTURE_STA=0;        //通道2输入捕获状态
  • u16        TIM5CH2_CAPTURE_VAL;        //通道2输入捕获值
  • u8  TIM5CH3_CAPTURE_STA=0;        //通道3输入捕获状态
  • u16        TIM5CH3_CAPTURE_VAL;        //通道3输入捕获值
  • u8  TIM5CH4_CAPTURE_STA=0;        //通道4输入捕获状态
  • u16        TIM5CH4_CAPTURE_VAL;        //通道4输入捕获值
  • u32 tempup1=0;//捕获总高电平的时间us单位
  • u32 tempup2=0;//捕获总高电平的时间us单位
  • u32 tempup3=0;//捕获总高电平的时间us单位
  • u32 tempup4=0;//捕获总高电平的时间us单位
  • unsigned int pwmout1,pwmout2,pwmout3,pwmout4;                                 //输出占空比
  • void TIM5_Cap_Init(u16 arr,u16 psc)
  • {
  •         GPIO_InitTypeDef GPIO_InitStructure;
  •         TIM_TimeBaseInitTypeDef  TIM_TimeBaseStructure;
  •         TIM_ICInitTypeDef  TIM5_ICInitStructure;
  •         RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM5, ENABLE);        //使能TIM5时钟
  •          RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);  //使能GPIOA时钟
  •         GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;  //PA0 清除之前设置
  •         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPD; //PA0 输入
  •         GPIO_Init(GPIOA, &GPIO_InitStructure);
  •         GPIO_ResetBits(GPIOA,GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3);
  •         //初始化定时器5 TIM5
  •         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(TIM5, &TIM_TimeBaseStructure); //根据TIM_TimeBaseInitStruct中指定的参数初始化TIMx的时间基数单位
  •         //初始化TIM5输入捕获参数
  •         TIM5_ICInitStructure.TIM_Channel = TIM_Channel_1; //CC1S=01         选择输入端 IC1映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;        //上升沿捕获
  •           TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;         //配置输入分频,不分频
  •           TIM5_ICInitStructure.TIM_ICFilter = 0x00;//IC1F=0000 配置输入滤波器 不滤波
  •           TIM_ICInit(TIM5, &TIM5_ICInitStructure);
  •         TIM5_ICInitStructure.TIM_Channel = TIM_Channel_2; //CC1S=01         选择输入端 IC1映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;        //上升沿捕获
  •           TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;         //配置输入分频,不分频
  •           TIM5_ICInitStructure.TIM_ICFilter = 0x00;//IC1F=0000 配置输入滤波器 不滤波
  •           TIM_ICInit(TIM5, &TIM5_ICInitStructure);
  •         TIM5_ICInitStructure.TIM_Channel = TIM_Channel_3; //CC1S=01         选择输入端 IC1映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;        //上升沿捕获
  •           TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;         //配置输入分频,不分频
  •           TIM5_ICInitStructure.TIM_ICFilter = 0x00;//IC1F=0000 配置输入滤波器 不滤波
  •           TIM_ICInit(TIM5, &TIM5_ICInitStructure);
  •         TIM5_ICInitStructure.TIM_Channel = TIM_Channel_4; //CC1S=01         选择输入端 IC1映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPolarity = TIM_ICPolarity_Rising;        //上升沿捕获
  •           TIM5_ICInitStructure.TIM_ICSelection = TIM_ICSelection_DirectTI; //映射到TI1上
  •           TIM5_ICInitStructure.TIM_ICPrescaler = TIM_ICPSC_DIV1;         //配置输入分频,不分频
  •           TIM5_ICInitStructure.TIM_ICFilter = 0x00;//IC1F=0000 配置输入滤波器 不滤波
  •           TIM_ICInit(TIM5, &TIM5_ICInitStructure);
  •         TIM_ITConfig(TIM5,TIM_IT_Update|TIM_IT_CC1|TIM_IT_CC2|TIM_IT_CC3|TIM_IT_CC4,ENABLE);//允许更新中断 ,允许CC1IE捕获中断
  •            TIM_Cmd(TIM5,ENABLE );         //使能定时器5
  • }
  • void get_pulsewidth(void)                                         //计算捕获的高电平时间
  • {
  • /*-------------------------------------------获取通道1高电平-----------------------------------*/
  •                         if(TIM5CH1_CAPTURE_STA&0X80)        //CH1 成功捕获到了一次上升沿
  •                     {
  •                            tempup1=TIM5CH1_CAPTURE_STA&0X3F;
  •                            tempup1*=65536;//溢出时间总和
  •                            tempup1+=TIM5CH1_CAPTURE_VAL;//得到总的高电平时间
  •                            pwmout1=tempup1/4;
  •                            TIM5CH1_CAPTURE_STA=0;//开启下一次捕获
  • /*--------------------------------------pwm输出限制-----------------------------------------*/
  •         //                        if(pwmout1<250)        pwmout1=250;
  •         //                        if(pwmout1>420)        pwmout1=420;                                                //油门限制
  • /*---------------------------------------------------------------------------------------------*/
  •                     }
  • /*-------------------------------------------获取通道2高电平-----------------------------------*/
  •                         if(TIM5CH2_CAPTURE_STA&0X80)             //CH2成功捕获到了一次上升沿
  •                     {
  •                            tempup2=TIM5CH2_CAPTURE_STA&0X3F;
  •                            tempup2*=65536;//溢出时间总和
  •                            tempup2+=TIM5CH2_CAPTURE_VAL;//得到总的高电平时间
  •                            pwmout2=tempup2/4;
  •                            TIM5CH2_CAPTURE_STA=0;//开启下一次捕获
  •                     }
  • /*-------------------------------------------获取通道3高电平-----------------------------------*/
  •                         if(TIM5CH3_CAPTURE_STA&0X80)          //CH3 成功捕获到了一次上升沿
  •                     {
  •                            tempup3=TIM5CH3_CAPTURE_STA&0X3F;
  •                            tempup3*=65536;//溢出时间总和
  •                            tempup3+=TIM5CH3_CAPTURE_VAL;//得到总的高电平时间
  •                            pwmout3=tempup3/4;
  •                            TIM5CH3_CAPTURE_STA=0;//开启下一次捕获
  •                     }
  • /*-------------------------------------------获取通道4高电平-----------------------------------*/
  •                         if(TIM5CH4_CAPTURE_STA&0X80)             //CH4 成功捕获到了一次上升沿
  •                     {
  •                            tempup4=TIM5CH4_CAPTURE_STA&0X3F;
  •                            tempup4*=65536;//溢出时间总和
  •                            tempup4+=TIM5CH4_CAPTURE_VAL;//得到总的高电平时间
  •                            pwmout4=tempup4/4;
  •                            TIM5CH4_CAPTURE_STA=0;//开启下一次捕获
  •                     }
  • }
  • //定时器5中断服务程序
  • void TIM5_IRQHandler(void)
  • {
  •    /*                                                                  */
  •          if((TIM5CH2_CAPTURE_STA&0X80)==0)//CH2还未成功捕获
  •         {
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)
  •                 {
  •                         if(TIM5CH2_CAPTURE_STA&0X40)//已经捕获到高电平了
  •                         {
  •                                 if((TIM5CH2_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
  •                                 {
  •                                         TIM5CH2_CAPTURE_STA|=0X80;//标记成功捕获了一次
  •                                         TIM5CH2_CAPTURE_VAL=0XFFFF;
  •                                 }
  •                                 else TIM5CH2_CAPTURE_STA++;
  •                         }
  •                 }
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_CC2) != RESET)//捕获1发生捕获事件
  •                 {
  •                         if(TIM5CH2_CAPTURE_STA&0X40)                //捕获到一个下降沿
  •                         {
  •                                 TIM5CH2_CAPTURE_STA|=0X80;                //标记成功捕获到一次上升沿
  •                                 TIM5CH2_CAPTURE_VAL=TIM_GetCapture2(TIM5);
  •                                    TIM_OC2PolarityConfig(TIM5,TIM_ICPolarity_Rising); //CC1P=0 设置为上升沿捕获
  •                         }
  •                         else                                                                  //还未开始,第一次捕获上升沿
  •                         {
  •                                 TIM5CH2_CAPTURE_STA=0;                        //清空
  •                                 TIM5CH2_CAPTURE_VAL=0;
  •                                  TIM_SetCounter(TIM5,0);
  •                                 TIM5CH2_CAPTURE_STA|=0X40;                //标记捕获到了上升沿
  •                                    TIM_OC2PolarityConfig(TIM5,TIM_ICPolarity_Falling);                //CC1P=1 设置为下降沿捕获
  •                         }
  •                 }
  •          }
  •          if((TIM5CH1_CAPTURE_STA&0X80)==0)//CH1 还未成功捕获
  •         {
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)
  •                 {
  •                         if(TIM5CH1_CAPTURE_STA&0X40)//已经捕获到高电平了
  •                         {
  •                                 if((TIM5CH1_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
  •                                 {
  •                                         TIM5CH1_CAPTURE_STA|=0X80;//标记成功捕获了一次
  •                                         TIM5CH1_CAPTURE_VAL=0XFFFF;
  •                                 }
  •                                 else TIM5CH1_CAPTURE_STA++;
  •                         }
  •                 }
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_CC1) != RESET)//捕获1发生捕获事件
  •                 {
  •                         if(TIM5CH1_CAPTURE_STA&0X40)                //捕获到一个下降沿
  •                         {
  •                                 TIM5CH1_CAPTURE_STA|=0X80;                //标记成功捕获到一次上升沿
  •                                 TIM5CH1_CAPTURE_VAL=TIM_GetCapture1(TIM5);
  •                                    TIM_OC1PolarityConfig(TIM5,TIM_ICPolarity_Rising); //CC1P=0 设置为上升沿捕获
  •                         }
  •                         else                                                                  //还未开始,第一次捕获上升沿
  •                         {
  •                                 TIM5CH1_CAPTURE_STA=0;                        //清空
  •                                 TIM5CH1_CAPTURE_VAL=0;
  •                                  TIM_SetCounter(TIM5,0);
  •                                 TIM5CH1_CAPTURE_STA|=0X40;                //标记捕获到了上升沿
  •                                    TIM_OC1PolarityConfig(TIM5,TIM_ICPolarity_Falling);                //CC1P=1 设置为下降沿捕获
  •                         }
  •                         }
  •          }
  •          if((TIM5CH3_CAPTURE_STA&0X80)==0)//CH3还未成功捕获
  •         {
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)
  •                 {
  •                         if(TIM5CH3_CAPTURE_STA&0X40)//已经捕获到高电平了
  •                         {
  •                                 if((TIM5CH3_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
  •                                 {
  •                                         TIM5CH3_CAPTURE_STA|=0X80;//标记成功捕获了一次
  •                                         TIM5CH3_CAPTURE_VAL=0XFFFF;
  •                                 }
  •                                 else TIM5CH3_CAPTURE_STA++;
  •                         }
  •                 }
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_CC3) != RESET)//捕获1发生捕获事件
  •                 {
  •                         if(TIM5CH3_CAPTURE_STA&0X40)                //捕获到一个下降沿
  •                         {
  •                                 TIM5CH3_CAPTURE_STA|=0X80;                //标记成功捕获到一次上升沿
  •                                 TIM5CH3_CAPTURE_VAL=TIM_GetCapture3(TIM5);
  •                                    TIM_OC3PolarityConfig(TIM5,TIM_ICPolarity_Rising); //CC1P=0 设置为上升沿捕获
  •                         }else                                                                  //还未开始,第一次捕获上升沿
  •                         {
  •                                 TIM5CH3_CAPTURE_STA=0;                        //清空
  •                                 TIM5CH3_CAPTURE_VAL=0;
  •                                  TIM_SetCounter(TIM5,0);
  •                                 TIM5CH3_CAPTURE_STA|=0X40;                //标记捕获到了上升沿
  •                                    TIM_OC3PolarityConfig(TIM5,TIM_ICPolarity_Falling);                //CC1P=1 设置为下降沿捕获
  •                         }
  •                         }
  •          }
  •          if((TIM5CH4_CAPTURE_STA&0X80)==0)//CH4 还未成功捕获
  •         {
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_Update) != RESET)
  •                 {
  •                         if(TIM5CH4_CAPTURE_STA&0X40)//已经捕获到高电平了
  •                         {
  •                                 if((TIM5CH4_CAPTURE_STA&0X3F)==0X3F)//高电平太长了
  •                                 {
  •                                         TIM5CH4_CAPTURE_STA|=0X80;//标记成功捕获了一次
  •                                         TIM5CH4_CAPTURE_VAL=0XFFFF;
  •                                 }
  •                                 else TIM5CH4_CAPTURE_STA++;
  •                         }
  •                 }
  •                 if (TIM_GetITStatus(TIM5, TIM_IT_CC4) != RESET)//捕获1发生捕获事件
  •                 {
  •                         if(TIM5CH4_CAPTURE_STA&0X40)                //捕获到一个下降沿
  •                         {
  •                                 TIM5CH4_CAPTURE_STA|=0X80;                //标记成功捕获到一次上升沿
  •                                 TIM5CH4_CAPTURE_VAL=TIM_GetCapture4(TIM5);
  •                                    TIM_OC4PolarityConfig(TIM5,TIM_ICPolarity_Rising); //CC1P=0 设置为上升沿捕获
  •                         }
  •                         else                                                                  //还未开始,第一次捕获上升沿
  • ……………………
  • …………限于本文篇幅 余下代码请从电子发烧友下载附件…………



所有资料提供下载:
        mwc基于stm32的飞控代码.rar
  
        毕设设计:基于STM32的四轴飞行器设计最终代码.rar

更多回帖

发帖
×
20
完善资料,
赚取积分