zoukankan      html  css  js  c++  java
  • 《SLAM机器人基础教程》第三章 单片机与STM32:电机码盘实现里程计实验

    3.12节 电机码盘实现里程计实验

    本节介绍里程计如何实现

    a.实验准备:码盘电机,USB转串口模块,ST-Llink下载器,CHEAPX机器人控制板

    b.实验目的:STM32实现里程计的数据采集

    c.相关知识点:

    里程计,指的是机器人行走的距离,通过统计码盘脉冲值转化而来。

    d.编程及运行

    (1)初始化

    void initEncoder(void)
    {
        //LFT ENC A:PB7 B:PB6
        //RGT ENC A:PB5 B:PB4
        
        GPIO_InitTypeDef GPIO_InitStruct;
        EXTI_InitTypeDef EXTI_InitStruct;
         NVIC_InitTypeDef NVIC_InitStruct;
        
        //Config Clock 配置时钟
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO|RCC_APB2Periph_GPIOB, ENABLE);
    
        GPIO_InitStruct.GPIO_Pin = GPIO_Pin_7 | GPIO_Pin_6| GPIO_Pin_5| GPIO_Pin_4;
      GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IPU;//上拉输入
      GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz;
      GPIO_Init(GPIOB, &GPIO_InitStruct);//传入结构体GPIO_InitStruct,配置GPIO
        
        //Config EXTILine 配置外部中断线
        //PB7
      GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource7);
      EXTI_InitStruct.EXTI_Line=EXTI_Line7;                         //外部中断线7
      EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;           //中断模式
      EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling;//上升沿和下降沿都触发
      EXTI_InitStruct.EXTI_LineCmd = ENABLE;                     //使能中断线
      EXTI_Init(&EXTI_InitStruct);                               //传入结构EXTI_InitStruct,配置外部中断
        NVIC_InitStruct.NVIC_IRQChannel = EXTI9_5_IRQn;                 //外部中断通道9-5    
        NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x02;     //抢占优先级
        NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x03;                 //子优先级        
        NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;                             //使能中断            
        NVIC_Init(&NVIC_InitStruct);                               //传入结构NVIC_InitStruct,配置NVIC
        //PB6
        GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource6);
      EXTI_InitStruct.EXTI_Line=EXTI_Line6;                         //外部中断线6
      EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;           //中断模式    
      EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling;//上升沿和下降沿都触发
      EXTI_InitStruct.EXTI_LineCmd = ENABLE;                     //使能中断线
      EXTI_Init(&EXTI_InitStruct);                               //传入结构EXTI_InitStruct,配置外部中断
        NVIC_InitStruct.NVIC_IRQChannel = EXTI9_5_IRQn;                 //外部中断通道    9-5
        NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x02;     //抢占优先级
        NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x03;                 //子优先级    
        NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;               //使能中断                                
        NVIC_Init(&NVIC_InitStruct);                               //传入结构NVIC_InitStruct,配置NVIC
        //PB5
        GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource5);
      EXTI_InitStruct.EXTI_Line=EXTI_Line5;                         //外部中断线5
      EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;           //中断模式    
      EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling;//上升沿和下降沿都触发
      EXTI_InitStruct.EXTI_LineCmd = ENABLE;                     //使能中断线
      EXTI_Init(&EXTI_InitStruct);                               //传入结构EXTI_InitStruct,配置外部中断
        NVIC_InitStruct.NVIC_IRQChannel = EXTI9_5_IRQn;                 //外部中断通道9-5
        NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x02;     //抢占优先级
        NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x03;                 //子优先级        
        NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;                             //使能中断            
        NVIC_Init(&NVIC_InitStruct);                               //传入结构NVIC_InitStruct,配置NVIC
        //PB4
        GPIO_EXTILineConfig(GPIO_PortSourceGPIOB,GPIO_PinSource4);
      EXTI_InitStruct.EXTI_Line=EXTI_Line4;                         //外部中断线4
      EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt;           //中断模式    
      EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling;//上升沿和下降沿都触发
      EXTI_InitStruct.EXTI_LineCmd = ENABLE;                     //使能中断线
      EXTI_Init(&EXTI_InitStruct);                               //传入结构EXTI_InitStruct,配置外部中断
        NVIC_InitStruct.NVIC_IRQChannel = EXTI4_IRQn;                   //外部中断通道4    
        NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x02;     //抢占优先级
        NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x03;                 //子优先级    
        NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE;                             //使能中断        
        NVIC_Init(&NVIC_InitStruct);                               //传入结构NVIC_InitStruct,配置NVIC      
    
    }

    (2)更新函数

    void updateEncoder(u8 id)
    {
        static u32 lastTimestampUS[2]={0,0};
        static float  unCountEncoder[2]={0,0};
        u32 TimestampUS=0;
        u32 deltaTimeUS=0;
        float deltaEncoderMM=0;
        
        //Timestamp时间戳,用于计算速度
        TimestampUS=9000000-SysTick->VAL;
        if(TimestampUS<lastTimestampUS[id])deltaTimeUS=TimestampUS+9000000-lastTimestampUS[id];
        else deltaTimeUS=TimestampUS-lastTimestampUS[id];    
        lastTimestampUS[id]=TimestampUS;
        
        
        //码盘同时输出2路相差90度相位的方波如下,因此码盘值应当只相差2。
      /*           _   _   _   _   _   _   _   _                          
        __________| |_| |_| |_| |_| |_| |_| |_| |_____________________________
                  _   _   _   _   _   _   _   _          
      ___________| |_| |_| |_| |_| |_| |_| |_| |____________________________
        */
        if(Encoder[id][0]>Encoder[id][1]+2)Encoder[id][1]=Encoder[id][0];
        else if(Encoder[id][1]>Encoder[id][0]+2)Encoder[id][0]=Encoder[id][1];
        else;
        
        
        deltaEncoder[id]=Encoder[id][0]+Encoder[id][1]+unCountEncoder[id];
        
        //odom里程计
        deltaEncoderMM=deltaEncoder[id]/encoder2mm;//码盘脉冲计数值转换为距离mm;encoder2mm = 轮子转一圈的码盘计数值/轮子周长 = 1560 / 204 = 7.6470588235294117647058823529412
        unCountEncoder[id]=deltaEncoder[id]-deltaEncoderMM*encoder2mm;   //计算由于精度原因未统计的值
        _EncoderUM[id]=_EncoderUM[id]+_motorDir[id]*deltaEncoderMM*1000; //计算总里程
        
        //caculate spd(unit: mm/s)
        if(deltaTimeUS>0)
        {
            MotorActualSpd[id] = deltaEncoderMM*(9000000/deltaTimeUS); //deltaTimeUS的单位是1/9us,转化为s,需要乘以9000000
        }
        if(id==0)printf("左码盘:%2u个;v=%3dmm/s,dt:%2u毫秒;",deltaEncoder[id],(int)MotorActualSpd[id],deltaTimeUS/9000);
        if(id==1)printf("右码盘:%2u个;v=%3dmm/s,dt:%2u毫秒.
    ",deltaEncoder[id],(int)MotorActualSpd[id],deltaTimeUS/9000);
        
        //清零
        Encoder[id][0]=0;
        Encoder[id][1]=0;
    }

    (3)主函数

    //电机码盘实现里程计实验
    int main(void)
    {
        
         NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); //PriorityGroupConfig NVIC中断分组设置 组2(2位抢占优先级,2位响应优先级)
        initDebugSerial(500000);//初始化调试串口USART1,波特率500000
        initSysTick();//初始化滴答定时器和TIM4定时器
        showVersion();//显示版本
        
        initMoter();  //初始化电机
        initEncoder();
        
        //参数:左电机速度,左电机方向,右电机速度,右电机方向
        //速度值范围[4000,30000],值过小电机可能会不转
        //方向:参照小车朝向,0往后转,1向前转
        moterControl(5000,1,5000,1); 
        
        while(1)
        {
            updateEncoder(0);
            updateEncoder(1);
            delay_ms(10);
        }
        
    }

    (4)实验结果

  • 相关阅读:
    云级Keyvalue数据库大比较
    如何诊断ORA125XX连接问题
    Oracle Internal Event:10235 Heap Checking诊断事件
    Oracle Database 11g R2 在Oracle Linux 6上获得认证
    一张图帮你搞懂oracle UGA User Global Area的概念
    Oracle Buffer Cache Chain图解
    ORA04036: 实例使用的 PGA 内存超出 PGA_AGGREGATE_LIMIT
    如何诊断ASMLIB故障
    Oracle RDBMS Server 11gR2 Preinstall RPM For Oracle Linux 6
    图解Oracle Buffer Cache LRU List
  • 原文地址:https://www.cnblogs.com/Baron-Lu/p/13378790.html
Copyright © 2011-2022 走看看