zoukankan      html  css  js  c++  java
  • [AVR]使用AVR单片机驱动舵机

    最近参加了三系举办的小车比赛(好像叫什么"驭远杯")。领导要求我驱动3-4个舵机。研究了几日,总算折腾出一个方案..、


    1.舵机驱动的基本原理

      (可以参考http://blog.sina.com.cn/s/blog_8240cbef01018hu1.html

      "控制信号由接收机的通道进入信号调制芯片,获得直流偏置电压。它内部有一个基准电路,产生周期为20ms,宽度为1.5ms的基准信号,将获得的直流偏置电压与电位器的电压比较,获得电压差输出。最后,电压差的正负输出到电机驱动芯片决定电机的正反转。当电机转速一定时,通过级联减速齿轮带动电位器旋转,使得电压差为0,电机停止转动。"

    简单的来讲,就是输出一个周期为20Ms,不同的占空比对应舵机转过不同的角度。

    难点主要在于

    • 舵机控制信号需要保持,这样就比用脉冲控制步进电机要复杂一些。
    • 你需要保持多路PWM,并且要随时调节占空比来获得要求的角度

    2.实现思路

    网上有用工作在相频修正PWM模式下的T1来产生信号,这样虽然十分精确,然而并不太好实现多路控制.(至少我是没想出来,如果有高手知道怎么做,还望多多指教)

    我决定采用以下方法:

    • 将 20ms 等分成240份,这样一份是20000/240 us //分成240份的原因是这样可以算出整数值得TCNT1
    • 将T1配置为溢出中断模式,每20000/240 us溢出一次
    • 中断服务程序更新TCNT1的值,维护一组变量,产生信号。

    这样做的优点是方便了多路控制。虽然我只控制了四路舵机,稍加修改就可以控制更多..

    然而中断服务程序中维护变量时,产生的微小误差会累加,这样不可避免的会产生较大误差。直接采用计算值肯定不行,最后需要修正。

    3.代码

    代码还是相当的不成熟...愿各位高手多多指教.

    通过传入一个指针给Servo_AngelPWM实现四个舵机的角度控制

    #include <util/delay.h>
    #include <stdint.h>
    #ifndef SERVO_CONTROL_H
    #define SERVO_CONTROL_H
    #define Servo1  PB7
    #define Servo1_1 PORTB|=_BV(Servo1)
    #define Servo1_0 PORTB&=~_BV(Servo1)
    #define Servo2  PB6
    #define Servo2_1 PORTB|=_BV(Servo2)
    #define Servo2_0 PORTB&=~_BV(Servo2)
    #define Servo3  PB5
    #define Servo3_1 PORTB|=_BV(Servo2)
    #define Servo3_0 PORTB&=~_BV(Servo2)
    #define Servo4  PB4
    #define Servo4_1 PORTB|=_BV(Servo2)
    #define Servo4_0 PORTB&=~_BV(Servo2)
    #define to_us(x) (((x/180.0)*2.0+0.5)*1000)
    uint32_t Servo_Flag[4];
    uint32_t Servo_Cflag=0;
    void Servo_AngelPWM(char *angel)
    {
        for(int i=0;i<4;i++)
            Servo_Flag[i]=to_us(angel[i])*3/250;
        TIMSK|=_BV(TOIE1);//开启TC1中断 
    }
    ISR(TIMER1_OVF_vect)
    {
        TIMSK&=~_BV(TOIE1);//关闭TC1中断               //1
        TCNT1=64525;//65535-(1000+10)这个10加的有讲究 //3
        Servo_Cflag++;                                  //4
        if(Servo_Cflag>=239)                          //5
        {
            Servo1_1;
            Servo2_1;
            Servo3_1;
            Servo4_1;
            Servo_Cflag=0;
        }
        else if(Servo_Flag[0]==Servo_Cflag)Servo1_0;  //6
        else if(Servo_Flag[1]==Servo_Cflag)Servo2_0;  //7
        else if(Servo_Flag[2]==Servo_Cflag)Servo3_0;  //8
        else if(Servo_Flag[3]==Servo_Cflag)Servo4_0;  //9
        TIMSK|=_BV(TOIE1);//开启TC1中断                //10
    }
    #endif

    4.后记

    给TCNT1赋计算值,也就是65535-1000=64525时,产生的信号大约是47HZ。如我所料,准确性比较差。

    然后尝试根据分析语句来修正TCNT1的初值,可以从我的注释看出...修正完后大约是48hz,还是不太准。

    最后直接上示波器微调了...当TCNT1为64569(修正值34)时,如上图,产生了比较准确的驱动信号(45度和90度)。

    (所以说学会汇编还是很重要的...有时间一定要研究研究)

  • 相关阅读:
    uva 11080(二分图染色)
    poj 3255(次短路)
    uva 707(记忆化搜索)
    uva 436(floyd变形)
    uva 11748(求可达矩阵)
    uva 11573(bfs)
    Codeforces Round #226 (Div. 2) 解题报告
    uva 11354(最小瓶颈路--多组询问 MST+LCA倍增)
    uva 534(最小瓶颈路)
    uva 538(简单图论 入度出度)
  • 原文地址:https://www.cnblogs.com/cn-lhc/p/5997855.html
Copyright © 2011-2022 走看看