找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 4758|回复: 1
打印 上一主题 下一主题
收起左侧

STM32+PCA9685(12位精度16通道PWM波输出)的程序源码

[复制链接]
跳转到指定楼层
楼主
ID:316013 发表于 2018-5-9 11:28 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
1.硬件

PCA9685是一款基于IIC总线通信的12位精度16通道PWM波输出的芯片,该芯片最初由NXP推出时主要面向LED开关调光,但就目前国内的形式来看,好像在被Arduino在舵机控制领域使用的更广泛。

该模块由于主要活跃在Aruino周边,所以在使用Arduino开发其底层驱动库是十分完善的,但对于单片机开发人员就不太友好了,需要自行根据用户手册在单片机上编写底层的驱动。

        其地址的分配是通过模块右上方的短接焊盘来确定的,从A0-A5表示地址的最低位到最高位,也就是最多可级联2^5=32个模块,地址为: 1+A5+A4+A3+A2+A1+A0+rw。如果不用短接的话Ax=0;短接的话Ax=1;rw为写的话rw=0;rw为读的话rw=1;所以写入数据不做短接则地址应该为1000 0000 =0x80,许多人包括淘宝商家误以为默认地址是0x40,坑爹啊。
        


2.IIC通信

SCL接PH4,SDA接PH5,VCC接3.3V,GND接,V+单纯只是供电,在驱动大功率舵机时不接,但是外接电源的两个端口要接上正负极,否则无法供电。


STM32单片机源程序如下:

  1. #include "stm32f10x_lib.h"
  2. #include "userconfig.h"
  3. #include "appdefine.h"
  4. #include "appfuns.h"
  5. #include "appvar.h"
  6. #include "appfiles.h"

  7. u8 mysendbuff[100];
  8. u8 HandShakeToMaster[7] = {0xAA, 0xBB, 0x00, 0x00, 0x00, 0x00, 0x65};//握手
  9. u8 WaterLowLine[7] = {0xAA, 0xEE, 0x00, 0x00, 0x00, 0x00, 0x98};  //请加水
  10. u8 TurnOffComputer[7] = {0xAA, 0xFF, 0x00, 0x00,0x00,0x00,0xA9};//停电关机


  11. void ProcessUSART(void){
  12.   USARTRecBuff *u;
  13.   SystermTask       *s;
  14.   u16 u16tempreceve;
  15.   MuxData temp;
  16.   u16 Motordata,Motorline;
  17.    
  18.   u = GetUSARTRecAddress();
  19.   s = GetTaskAddress();
  20.   if(u->COMState == COMBUSY){//如有完整帧接收
  21.     /*通讯协议*/
  22.     if(u->RxBuff[0] == 0xAA){
  23.       u16tempreceve = GenCheck(&u->RxBuff[0],6);
  24.      if(u->RxBuff[6] != u16tempreceve){
  25.        Send2USART2("err",3);
  26.      }
  27.      else{
  28.       Send2USART2((u8*)HandShakeToMaster,7);
  29.        DelayX10us(50);
  30.       switch(u->RxBuff[1]){
  31.       case 0x03:
  32.         StopAllMotor();
  33.         temp. u8_data[0]= u->RxBuff[3] ;
  34.         temp. u8_data[1]= u->RxBuff[2] ;
  35.         Motordata = temp.u16_data[0];
  36.         temp. u8_data[2]= u->RxBuff[5] ;
  37.         temp. u8_data[3]= u->RxBuff[4];
  38.         Motorline =  temp.u16_data[1];
  39.         MotorControl(0x01,Motordata,Motorline);//开
  40.      //   MotorControl(0x01,u->RxBuff[3],u->RxBuff[5]);//开
  41.        // TurnOnELS();
  42.         s->MotorBit = 1;
  43.         s->Medecinestate = Medecine_Out_Testing;
  44.         break;
  45.       case 0x04:
  46.          MotorControl(0x00,u->RxBuff[3],u->RxBuff[5]);//关
  47.         break;
  48.       case 0x05:
  49.         StopAllMotor();         
  50.         break;
  51.       case 0x06:
  52.         break;
  53.       case 0x07:
  54.         TurnOnELS();
  55.         CheckElsOn();
  56.         break;
  57.       case 0x08:
  58.         TurnOffELS();
  59.         CheckElsOff();
  60.         break;
  61.       case 0x09:
  62.         CheckEls();
  63.         break;
  64.         
  65.       case 0x0E:
  66.         ControlOut1_On();
  67.         break;
  68.       case 0x0F:
  69.         ControlOut1_Off();
  70.         break;
  71.       default:
  72.         break;
  73.       }
  74.       
  75.      }
  76.     }
  77.     else if(u->RxBuff[0] == 0xFF){
  78.     //   ShowString (&(u->RxBuff[3]) ,u->RxBuff[2] ,202 ,80);
  79.       u->length = u->RxBuff[2];
  80.       u16tempreceve = GenCheck(&u->RxBuff[0],(u->RxBuff[2]+3));
  81.      if(u->RxBuff[u->length+3] != u16tempreceve){
  82.        Send2USART2("err",3);
  83.     }
  84.      else if(u->RxBuff[1] == 0x01){  
  85.        // ClearDisplay();
  86.         Choicepage(1);
  87.       ShowString (&(u->RxBuff[3]) ,u->RxBuff[2] ,202 ,100);
  88.      }
  89.     }
  90.     else if(u->RxBuff[0] == 0xEE){
  91.        u->length = u->RxBuff[2];
  92.       u16tempreceve = GenCheck(&u->RxBuff[0],(u->RxBuff[2]+3));
  93.      if(u->RxBuff[u->length+3] != u16tempreceve){
  94.        Send2USART2("err",3);
  95.     }
  96.      else if(u->RxBuff[1] == 0x12){     
  97.        ShowString (&(u->RxBuff[3]) ,u->RxBuff[2] ,202 ,80);   
  98.      }
  99.    
  100.     }
  101.     u->RxCount = 0;
  102.     u->COMState = COMIDLE;
  103.   }
  104. }

  105. void ProcessTask(void){

  106. Motor_Set_Ang(15,120);
  107.   delay_ms(250);
  108.   Motor_Set_Ang(15, 0);
  109.   delay_ms(250);
  110.   Motor_Set_OFF(15);
  111.   
  112.   Motor_Set_Ang(14,120);
  113.   delay_ms(250);
  114.   Motor_Set_Ang(14, 0);
  115.   delay_ms(250);
  116.   Motor_Set_OFF(14);
  117.   
  118.   Motor_Set_Ang(13,120);
  119.   delay_ms(250);
  120.   Motor_Set_Ang(13, 0);
  121.   delay_ms(250);
  122.   Motor_Set_OFF(13);

  123.   Motor_Set_Ang(12,120);
  124.   delay_ms(250);
  125.   Motor_Set_Ang(12, 0);
  126.   delay_ms(250);
  127.   Motor_Set_OFF(12);

  128.   Motor_Set_Ang(11,120);
  129.   delay_ms(250);
  130.   Motor_Set_Ang(11, 0);
  131.   delay_ms(250);
  132.   Motor_Set_OFF(11);

  133.   
  134.   Motor_Set_Ang(10,120);
  135.   delay_ms(250);
  136.   Motor_Set_Ang(10, 0);
  137.   delay_ms(250);
  138.   Motor_Set_OFF(10);

  139.   Motor_Set_Ang(9,120);
  140.   delay_ms(250);
  141.   Motor_Set_Ang(9, 0);
  142.   delay_ms(250);
  143.   Motor_Set_OFF(9);
  144.   
  145.   Motor_Set_Ang(8,120);
  146.   delay_ms(250);
  147.   Motor_Set_Ang(8, 0);
  148.   delay_ms(250);
  149.   Motor_Set_OFF(8);
  150.   
  151.   Motor_Set_Ang(7,120);
  152.   delay_ms(250);
  153.   Motor_Set_Ang(7, 0);
  154.   delay_ms(250);
  155.   Motor_Set_OFF(7);
  156.   
  157.   Motor_Set_Ang(6,120);
  158.   delay_ms(250);
  159.   Motor_Set_Ang(6, 0);
  160.   delay_ms(250);
  161.   Motor_Set_OFF(6);
  162.   
  163.   Motor_Set_Ang(5,120);
  164.   delay_ms(250);
  165.   Motor_Set_Ang(5, 0);
  166.   delay_ms(250);
  167.   Motor_Set_OFF(5);
  168.   
  169.   Motor_Set_Ang(4,120);
  170.   delay_ms(250);
  171.   Motor_Set_Ang(4, 0);
  172.   delay_ms(250);
  173.   Motor_Set_OFF(4);
  174.   
  175.   Motor_Set_Ang(3,120);
  176.   delay_ms(250);
  177.   Motor_Set_Ang(3, 0);
  178.   delay_ms(250);
  179.   Motor_Set_OFF(3);
  180.   
  181.   Motor_Set_Ang(2,120);
  182.   delay_ms(250);
  183.   Motor_Set_Ang(2, 0);
  184.   delay_ms(250);
  185.   Motor_Set_OFF(2);
  186.   
  187.   Motor_Set_Ang(1,120);
  188.   delay_ms(250);
  189.   Motor_Set_Ang(1, 0);
  190.   delay_ms(250);
  191.   Motor_Set_OFF(1);
  192.   
  193.   Motor_Set_Ang(0,120);
  194.   delay_ms(250);
  195.   Motor_Set_Ang(0, 0);
  196.   delay_ms(250);
  197. }

  198. void ProcessCtrl(void){
  199. }

  200. int main(void){

  201.   
  202.   SystermInit();
  203.   I2C_PCA9685_Init();
  204.   OE_Init();
  205.    
  206.   while(1){

  207.     ProcessUSART();
  208.     ProcessCtrl();
  209.     ProcessTask();
  210.   }
  211. }
复制代码

所有资料51hei提供下载:

SoftwareforPCA9685.rar (640.96 KB, 下载次数: 56)




评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 分享淘帖 顶 踩
回复

使用道具 举报

沙发
ID:352407 发表于 2019-1-16 14:13 | 只看该作者
资料很好
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表